From 08f179808e14cd48d05cd725e0d2996ffb03e83d Mon Sep 17 00:00:00 2001 From: Joe Robertson Date: Wed, 18 Dec 2024 10:18:57 -0700 Subject: [PATCH] Squashed 'resources/hpxml-measures/' changes from 5bad046428ef..01b93fb883cb 01b93fb883cb Latest results. b17f24ee7681 Fix ASHRAE 140 runs. bc1e205023d2 Fixes zero occupants specified for one unit in a whole MF building from being treated like zero occupants for every unit. 2d8bf015b621 Remove some unused code I noticed in tasks.rb [ci skip] 3b7dabd640cf Merge pull request #1895 from NREL/docs_water_heater 8068f153d261 Add Model.add_plant_loop() method. 2b64d9f800df Bugfix. f7e65bee01b5 Merge branch 'master' of https://github.com/NREL/OpenStudio-HPXML into docs_water_heater 926c3dedc72f Added ruby documentation to waterheater.rb. b5d8972e74c9 Bump version back to 1.10.0. 94d6687c2d4b Merge pull request #1891 from NREL/manual_j_more_improvements 59d9cdbac678 Update measure 172416f362ca Switch version to 1.9.1 for release. Update docs/readme/changelog. f206771b0454 Latest results. 5e6784b5d5fd Merge branch 'manual_j_more_improvements' of https://github.com/NREL/OpenStudio-HPXML into manual_j_more_improvements 72c55f7e7332 fix bug while cleaningup 024d5007bf29 Latest results. ee5e87e82aee Merge branch 'master' of https://github.com/NREL/OpenStudio-HPXML into manual_j_more_improvements 0d1232c0164a Merge branch 'manual_j_more_improvements' of https://github.com/NREL/OpenStudio-HPXML into manual_j_more_improvements ac134c96f3a0 3-23 crawlspace wall height change 09fb00ce9e02 Merge pull request #1892 from NREL/buildreshpxml_docs_version d968043c4861 Merge branch 'master' of https://github.com/NREL/OpenStudio-HPXML into buildreshpxml_docs_version 1066943a60dd Merge pull request #1894 from NREL/misc_refactor_cleanup d250e3386777 Latest results. 3dc5f7e3ddf2 Missed one more. dc4ecf707e8b Latest results. 13984347ef63 Merge branch 'misc_refactor_cleanup' of https://github.com/NREL/OpenStudio-HPXML into misc_refactor_cleanup 0d3207df51b7 Bugfix. d08d7b631112 Latest results. 3c7fcabee234 Revert change. c8573ecfc8a9 Bugfix. 37732581d283 Misc refactor/cleanup and minor bugfixes. Updates HPXML::has_fuels() to traverse the HPXML object for performance reasons (particularly important when running whole SFA/MF buildings). db788d3286fd Adds a test to ensure BuildResidentialHPXML measure docs links are updated when the OS-HPXML version is bumped. 08673f55c689 Latest results. 65d756929a5a fix typo c5831f52e616 minor improvements 271b4a14b320 Shrink docs image [ci skip] e6a3ccf2b784 Minor docs cleanup [ci skip] a2d97ef06e84 Fix formatting in docs [ci skip] 3d7eadebea5a Merge pull request #1888 from NREL/bump_version 4c2886fb6a73 Update to version 1.10.0 git-subtree-dir: resources/hpxml-measures git-subtree-split: 01b93fb883cb2883c7d17746f017ad4c83ef6d3a --- BuildResidentialHPXML/README.md | 404 ++++---- BuildResidentialHPXML/measure.rb | 198 ++-- BuildResidentialHPXML/measure.xml | 416 ++++---- BuildResidentialHPXML/resources/geometry.rb | 102 +- .../tests/test_build_residential_hpxml.rb | 375 ++++---- BuildResidentialScheduleFile/measure.xml | 6 +- .../resources/schedules.rb | 2 +- Changelog.md | 15 + HPXMLtoOpenStudio/measure.rb | 14 - HPXMLtoOpenStudio/measure.xml | 58 +- HPXMLtoOpenStudio/resources/airflow.rb | 97 +- HPXMLtoOpenStudio/resources/constructions.rb | 69 +- HPXMLtoOpenStudio/resources/defaults.rb | 654 ++++++++----- HPXMLtoOpenStudio/resources/energyplus.rb | 29 +- HPXMLtoOpenStudio/resources/geometry.rb | 18 +- .../resources/hotwater_appliances.rb | 191 ++-- HPXMLtoOpenStudio/resources/hpxml.rb | 146 +-- .../hpxml_schematron/EPvalidator.xml | 1 + HPXMLtoOpenStudio/resources/hvac.rb | 224 +++-- HPXMLtoOpenStudio/resources/hvac_sizing.rb | 167 ++-- HPXMLtoOpenStudio/resources/internal_gains.rb | 14 +- HPXMLtoOpenStudio/resources/lighting.rb | 7 +- HPXMLtoOpenStudio/resources/materials.rb | 58 +- HPXMLtoOpenStudio/resources/misc_loads.rb | 37 +- HPXMLtoOpenStudio/resources/model.rb | 34 +- HPXMLtoOpenStudio/resources/pv.rb | 26 +- HPXMLtoOpenStudio/resources/schedules.rb | 14 +- HPXMLtoOpenStudio/resources/utility_bills.rb | 10 +- HPXMLtoOpenStudio/resources/version.rb | 2 +- HPXMLtoOpenStudio/resources/waterheater.rb | 898 +++++++++--------- HPXMLtoOpenStudio/resources/xmlhelper.rb | 41 +- HPXMLtoOpenStudio/tests/test_airflow.rb | 20 + .../tests/test_hotwater_appliance.rb | 22 + HPXMLtoOpenStudio/tests/test_lighting.rb | 15 + HPXMLtoOpenStudio/tests/test_miscloads.rb | 63 +- HPXMLtoOpenStudio/tests/test_validation.rb | 648 ++++++------- README.md | 4 +- ReportSimulationOutput/measure.rb | 18 +- ReportSimulationOutput/measure.xml | 6 +- ReportUtilityBills/measure.rb | 18 +- ReportUtilityBills/measure.xml | 8 +- .../tests/test_report_utility_bills.rb | 13 +- docs/source/images/slab.png | Bin 150528 -> 51931 bytes docs/source/intro.rst | 8 + docs/source/testing_framework.rst | 3 +- docs/source/workflow_inputs.rst | 14 +- tasks.rb | 19 - workflow/hpxml_inputs.json | 4 - .../sample_files/base-location-detailed.xml | 6 + .../base-residents-0-runperiod-1-month.xml | 558 ----------- .../ACCA_Examples/Bob_Ross_Residence_3-22.xml | 6 +- .../ACCA_Examples/Bob_Ross_Residence_3-23.xml | 13 +- .../tests/base_results/results_acca_hvac.csv | 4 +- .../results_simulations_bills.csv | 1 - .../results_simulations_energy.csv | 1 - .../base_results/results_simulations_hvac.csv | 1 - .../results_simulations_loads.csv | 1 - .../base_results/results_simulations_misc.csv | 1 - 58 files changed, 2869 insertions(+), 2933 deletions(-) delete mode 100644 workflow/sample_files/base-residents-0-runperiod-1-month.xml diff --git a/BuildResidentialHPXML/README.md b/BuildResidentialHPXML/README.md index c322c00d0e..0c76275dc2 100644 --- a/BuildResidentialHPXML/README.md +++ b/BuildResidentialHPXML/README.md @@ -101,7 +101,7 @@ Specifies the unavailable period date ranges. Enter a date range like "Dec 15 - **Schedules: Unavailable Period Window Natural Ventilation Availabilities** -The availability of the natural ventilation schedule during unavailable periods. Valid choices are: regular schedule, always available, always unavailable. If multiple periods, use a comma-separated list. If not provided, the OS-HPXML default (see HPXML Unavailable Periods) is used. +The availability of the natural ventilation schedule during unavailable periods. Valid choices are: regular schedule, always available, always unavailable. If multiple periods, use a comma-separated list. If not provided, the OS-HPXML default (see HPXML Unavailable Periods) is used. - **Name:** ``schedules_unavailable_period_window_natvent_availabilities`` - **Type:** ``String`` @@ -112,7 +112,7 @@ The availability of the natural ventilation schedule during unavailable periods. **Simulation Control: Timestep** -Value must be a divisor of 60. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. +Value must be a divisor of 60. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. - **Name:** ``simulation_control_timestep`` - **Type:** ``Integer`` @@ -125,7 +125,7 @@ Value must be a divisor of 60. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. +Enter a date range like 'Jan 1 - Dec 31'. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. - **Name:** ``simulation_control_run_period`` - **Type:** ``String`` @@ -136,7 +136,7 @@ Enter a date range like 'Jan 1 - Dec 31'. If not provided, the OS-HPXML default **Simulation Control: Run Period Calendar Year** -This numeric field should contain the calendar year that determines the start day of week. If you are running simulations using AMY weather files, the value entered for calendar year will not be used; it will be overridden by the actual year found in the AMY weather file. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. +This numeric field should contain the calendar year that determines the start day of week. If you are running simulations using AMY weather files, the value entered for calendar year will not be used; it will be overridden by the actual year found in the AMY weather file. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. - **Name:** ``simulation_control_run_period_calendar_year`` - **Type:** ``Integer`` @@ -149,7 +149,7 @@ This numeric field should contain the calendar year that determines the start da **Simulation Control: Daylight Saving Enabled** -Whether to use daylight saving. If not provided, the OS-HPXML default (see HPXML Building Site) is used. +Whether to use daylight saving. If not provided, the OS-HPXML default (see HPXML Building Site) is used. - **Name:** ``simulation_control_daylight_saving_enabled`` - **Type:** ``Boolean`` @@ -160,7 +160,7 @@ Whether to use daylight saving. If not provided, the OS-HPXML default (see HPXML Building Site) is used. +Enter a date range like 'Mar 15 - Dec 15'. If not provided, the OS-HPXML default (see HPXML Building Site) is used. - **Name:** ``simulation_control_daylight_saving_period`` - **Type:** ``String`` @@ -171,7 +171,7 @@ Enter a date range like 'Mar 15 - Dec 15'. If not provided, the OS-HPXML default **Simulation Control: Temperature Capacitance Multiplier** -Affects the transient calculation of indoor air temperatures. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. +Affects the transient calculation of indoor air temperatures. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. - **Name:** ``simulation_control_temperature_capacitance_multiplier`` - **Type:** ``String`` @@ -182,7 +182,7 @@ Affects the transient calculation of indoor air temperatures. If not provided, t **Simulation Control: Defrost Model Type** -Research feature to select the type of defrost model. Use standard for default E+ defrost setting. Use advanced for an improved model that better accounts for load and energy use during defrost; using advanced may impact simulation runtime. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. +Research feature to select the type of defrost model. Use standard for default E+ defrost setting. Use advanced for an improved model that better accounts for load and energy use during defrost; using advanced may impact simulation runtime. If not provided, the OS-HPXML default (see HPXML Simulation Control) is used. - **Name:** ``simulation_control_defrost_model_type`` - **Type:** ``Choice`` @@ -221,7 +221,7 @@ Research feature to model capacity increment of multi-stage heat pump backup sys **Site: Type** -The type of site. If not provided, the OS-HPXML default (see HPXML Site) is used. +The type of site. If not provided, the OS-HPXML default (see HPXML Site) is used. - **Name:** ``site_type`` - **Type:** ``Choice`` @@ -234,7 +234,7 @@ The type of site. If not provided, the OS-HPXML default (see HPXML Site) is used. +Presence of nearby buildings, trees, obstructions for infiltration model. If not provided, the OS-HPXML default (see HPXML Site) is used. - **Name:** ``site_shielding_of_home`` - **Type:** ``Choice`` @@ -247,7 +247,7 @@ Presence of nearby buildings, trees, obstructions for infiltration model. If not **Site: Soil and Moisture Type** -Type of soil and moisture. This is used to inform ground conductivity and diffusivity. If not provided, the OS-HPXML default (see HPXML Site) is used. +Type of soil and moisture. This is used to inform ground conductivity and diffusivity. If not provided, the OS-HPXML default (see HPXML Site) is used. - **Name:** ``site_soil_and_moisture_type`` - **Type:** ``Choice`` @@ -310,7 +310,7 @@ City/municipality of the home address. **Site: State Code** -State code of the home address. If not provided, the OS-HPXML default (see HPXML Site) is used. +State code of the home address. If not provided, the OS-HPXML default (see HPXML Site) is used. - **Name:** ``site_state_code`` - **Type:** ``Choice`` @@ -334,7 +334,7 @@ Zip code of the home address. Either this or the Weather Station: EnergyPlus Wea **Site: Time Zone UTC Offset** -Time zone UTC offset of the home address. Must be between -12 and 14. If not provided, the OS-HPXML default (see HPXML Site) is used. +Time zone UTC offset of the home address. Must be between -12 and 14. If not provided, the OS-HPXML default (see HPXML Site) is used. - **Name:** ``site_time_zone_utc_offset`` - **Type:** ``Double`` @@ -347,7 +347,7 @@ Time zone UTC offset of the home address. Must be between -12 and 14. If not pro **Site: Elevation** -Elevation of the home address. If not provided, the OS-HPXML default (see HPXML Site) is used. +Elevation of the home address. If not provided, the OS-HPXML default (see HPXML Site) is used. - **Name:** ``site_elevation`` - **Type:** ``Double`` @@ -360,7 +360,7 @@ Elevation of the home address. If not provided, the OS-HPXML default (see HPXML Site) is used. +Latitude of the home address. Must be between -90 and 90. Use negative values for southern hemisphere. If not provided, the OS-HPXML default (see HPXML Site) is used. - **Name:** ``site_latitude`` - **Type:** ``Double`` @@ -373,7 +373,7 @@ Latitude of the home address. Must be between -90 and 90. Use negative values fo **Site: Longitude** -Longitude of the home address. Must be between -180 and 180. Use negative values for the western hemisphere. If not provided, the OS-HPXML default (see HPXML Site) is used. +Longitude of the home address. Must be between -180 and 180. Use negative values for the western hemisphere. If not provided, the OS-HPXML default (see HPXML Site) is used. - **Name:** ``site_longitude`` - **Type:** ``Double`` @@ -541,7 +541,7 @@ The number of bedrooms in the unit. **Geometry: Unit Number of Bathrooms** -The number of bathrooms in the unit. If not provided, the OS-HPXML default (see HPXML Building Construction) is used. +The number of bathrooms in the unit. If not provided, the OS-HPXML default (see HPXML Building Construction) is used. - **Name:** ``geometry_unit_num_bathrooms`` - **Type:** ``Integer`` @@ -554,7 +554,7 @@ The number of bathrooms in the unit. If not provided, the OS-HPXML default (see **Geometry: Unit Number of Occupants** -The number of occupants in the unit. If not provided, an *asset* calculation is performed assuming standard occupancy, in which various end use defaults (e.g., plug loads, appliances, and hot water usage) are calculated based on Number of Bedrooms and Conditioned Floor Area per ANSI/RESNET/ICC 301-2019. If provided, an *operational* calculation is instead performed in which the end use defaults are adjusted using the relationship between Number of Bedrooms and Number of Occupants from RECS 2015. +The number of occupants in the unit. If not provided, an *asset* calculation is performed assuming standard occupancy, in which various end use defaults (e.g., plug loads, appliances, and hot water usage) are calculated based on Number of Bedrooms and Conditioned Floor Area per ANSI/RESNET/ICC 301. If provided, an *operational* calculation is instead performed in which the end use defaults to reflect real-world data (where possible). - **Name:** ``geometry_unit_num_occupants`` - **Type:** ``Double`` @@ -593,7 +593,7 @@ Average distance from the floor to the ceiling. **Geometry: Unit Height Above Grade** -Describes the above-grade height of apartment units on upper floors or homes above ambient or belly-and-wing foundations. It is defined as the height of the lowest conditioned floor above grade and is used to calculate the wind speed for the infiltration model. If not provided, the OS-HPXML default (see HPXML Building Construction) is used. +Describes the above-grade height of apartment units on upper floors or homes above ambient or belly-and-wing foundations. It is defined as the height of the lowest conditioned floor above grade and is used to calculate the wind speed for the infiltration model. If not provided, the OS-HPXML default (see HPXML Building Construction) is used. - **Name:** ``geometry_unit_height_above_grade`` - **Type:** ``Double`` @@ -814,7 +814,7 @@ The distance between the unit and the neighboring building to the right (not inc **Neighbor: Front Height** -The height of the neighboring building to the front. If not provided, the OS-HPXML default (see HPXML Neighbor Building) is used. +The height of the neighboring building to the front. If not provided, the OS-HPXML default (see HPXML Neighbor Building) is used. - **Name:** ``neighbor_front_height`` - **Type:** ``Double`` @@ -827,7 +827,7 @@ The height of the neighboring building to the front. If not provided, the OS-HPX **Neighbor: Back Height** -The height of the neighboring building to the back. If not provided, the OS-HPXML default (see HPXML Neighbor Building) is used. +The height of the neighboring building to the back. If not provided, the OS-HPXML default (see HPXML Neighbor Building) is used. - **Name:** ``neighbor_back_height`` - **Type:** ``Double`` @@ -840,7 +840,7 @@ The height of the neighboring building to the back. If not provided, the OS-HPXM **Neighbor: Left Height** -The height of the neighboring building to the left. If not provided, the OS-HPXML default (see HPXML Neighbor Building) is used. +The height of the neighboring building to the left. If not provided, the OS-HPXML default (see HPXML Neighbor Building) is used. - **Name:** ``neighbor_left_height`` - **Type:** ``Double`` @@ -853,7 +853,7 @@ The height of the neighboring building to the left. If not provided, the OS-HPXM **Neighbor: Right Height** -The height of the neighboring building to the right. If not provided, the OS-HPXML default (see HPXML Neighbor Building) is used. +The height of the neighboring building to the right. If not provided, the OS-HPXML default (see HPXML Neighbor Building) is used. - **Name:** ``neighbor_right_height`` - **Type:** ``Double`` @@ -905,7 +905,7 @@ The type of floors. **Foundation Wall: Type** -The material type of the foundation wall. If not provided, the OS-HPXML default (see HPXML Foundation Walls) is used. +The material type of the foundation wall. If not provided, the OS-HPXML default (see HPXML Foundation Walls) is used. - **Name:** ``foundation_wall_type`` - **Type:** ``Choice`` @@ -918,7 +918,7 @@ The material type of the foundation wall. If not provided, the OS-HPXML default **Foundation Wall: Thickness** -The thickness of the foundation wall. If not provided, the OS-HPXML default (see HPXML Foundation Walls) is used. +The thickness of the foundation wall. If not provided, the OS-HPXML default (see HPXML Foundation Walls) is used. - **Name:** ``foundation_wall_thickness`` - **Type:** ``Double`` @@ -957,7 +957,7 @@ Whether the insulation is on the interior or exterior of the foundation wall. On **Foundation Wall: Insulation Distance To Top** -The distance from the top of the foundation wall to the top of the foundation wall insulation. Only applies to basements/crawlspaces. If not provided, the OS-HPXML default (see HPXML Foundation Walls) is used. +The distance from the top of the foundation wall to the top of the foundation wall insulation. Only applies to basements/crawlspaces. If not provided, the OS-HPXML default (see HPXML Foundation Walls) is used. - **Name:** ``foundation_wall_insulation_distance_to_top`` - **Type:** ``Double`` @@ -970,7 +970,7 @@ The distance from the top of the foundation wall to the top of the foundation wa **Foundation Wall: Insulation Distance To Bottom** -The distance from the top of the foundation wall to the bottom of the foundation wall insulation. Only applies to basements/crawlspaces. If not provided, the OS-HPXML default (see HPXML Foundation Walls) is used. +The distance from the top of the foundation wall to the bottom of the foundation wall insulation. Only applies to basements/crawlspaces. If not provided, the OS-HPXML default (see HPXML Foundation Walls) is used. - **Name:** ``foundation_wall_insulation_distance_to_bottom`` - **Type:** ``Double`` @@ -1100,7 +1100,7 @@ Width from slab edge inward of horizontal under-slab insulation. Enter 999 to sp **Slab: Thickness** -The thickness of the slab. Zero can be entered if there is a dirt floor instead of a slab. If not provided, the OS-HPXML default (see HPXML Slabs) is used. +The thickness of the slab. Zero can be entered if there is a dirt floor instead of a slab. If not provided, the OS-HPXML default (see HPXML Slabs) is used. - **Name:** ``slab_thickness`` - **Type:** ``Double`` @@ -1113,7 +1113,7 @@ The thickness of the slab. Zero can be entered if there is a dirt floor instead **Slab: Carpet Fraction** -Fraction of the slab floor area that is carpeted. If not provided, the OS-HPXML default (see HPXML Slabs) is used. +Fraction of the slab floor area that is carpeted. If not provided, the OS-HPXML default (see HPXML Slabs) is used. - **Name:** ``slab_carpet_fraction`` - **Type:** ``Double`` @@ -1126,7 +1126,7 @@ Fraction of the slab floor area that is carpeted. If not provided, the OS-HPXML **Slab: Carpet R-value** -R-value of the slab carpet. If not provided, the OS-HPXML default (see HPXML Slabs) is used. +R-value of the slab carpet. If not provided, the OS-HPXML default (see HPXML Slabs) is used. - **Name:** ``slab_carpet_r`` - **Type:** ``Double`` @@ -1152,7 +1152,7 @@ Assembly R-value for the ceiling (attic floor). **Roof: Material Type** -The material type of the roof. If not provided, the OS-HPXML default (see HPXML Roofs) is used. +The material type of the roof. If not provided, the OS-HPXML default (see HPXML Roofs) is used. - **Name:** ``roof_material_type`` - **Type:** ``Choice`` @@ -1165,7 +1165,7 @@ The material type of the roof. If not provided, the OS-HPXML default (see HPXML Roofs) is used. +The color of the roof. If not provided, the OS-HPXML default (see HPXML Roofs) is used. - **Name:** ``roof_color`` - **Type:** ``Choice`` @@ -1204,7 +1204,7 @@ The location of the radiant barrier in the attic. **Attic: Radiant Barrier Grade** -The grade of the radiant barrier in the attic. If not provided, the OS-HPXML default (see HPXML Roofs) is used. +The grade of the radiant barrier in the attic. If not provided, the OS-HPXML default (see HPXML Roofs) is used. - **Name:** ``radiant_barrier_grade`` - **Type:** ``Choice`` @@ -1230,7 +1230,7 @@ The type of walls. **Wall: Siding Type** -The siding type of the walls. Also applies to rim joists. If not provided, the OS-HPXML default (see HPXML Walls) is used. +The siding type of the walls. Also applies to rim joists. If not provided, the OS-HPXML default (see HPXML Walls) is used. - **Name:** ``wall_siding_type`` - **Type:** ``Choice`` @@ -1243,7 +1243,7 @@ The siding type of the walls. Also applies to rim joists. If not provided, the O **Wall: Color** -The color of the walls. Also applies to rim joists. If not provided, the OS-HPXML default (see HPXML Walls) is used. +The color of the walls. Also applies to rim joists. If not provided, the OS-HPXML default (see HPXML Walls) is used. - **Name:** ``wall_color`` - **Type:** ``Choice`` @@ -1386,7 +1386,7 @@ Ratio of window height to width. **Windows: Fraction Operable** -Fraction of windows that are operable. If not provided, the OS-HPXML default (see HPXML Windows) is used. +Fraction of windows that are operable. If not provided, the OS-HPXML default (see HPXML Windows) is used. - **Name:** ``window_fraction_operable`` - **Type:** ``Double`` @@ -1399,7 +1399,7 @@ Fraction of windows that are operable. If not provided, the OS-HPXML default (se **Windows: Natural Ventilation Availability** -For operable windows, the number of days/week that windows can be opened by occupants for natural ventilation. If not provided, the OS-HPXML default (see HPXML Windows) is used. +For operable windows, the number of days/week that windows can be opened by occupants for natural ventilation. If not provided, the OS-HPXML default (see HPXML Windows) is used. - **Name:** ``window_natvent_availability`` - **Type:** ``Integer`` @@ -1436,7 +1436,7 @@ Full-assembly NFRC solar heat gain coefficient. **Windows: Interior Shading Type** -Type of window interior shading. Summer/winter shading coefficients can be provided below instead. If neither is provided, the OS-HPXML default (see HPXML Interior Shading) is used. +Type of window interior shading. Summer/winter shading coefficients can be provided below instead. If neither is provided, the OS-HPXML default (see HPXML Interior Shading) is used. - **Name:** ``window_interior_shading_type`` - **Type:** ``Choice`` @@ -1449,7 +1449,7 @@ Type of window interior shading. Summer/winter shading coefficients can be provi **Windows: Winter Interior Shading Coefficient** -Interior shading coefficient for the winter season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see HPXML Interior Shading) is used. +Interior shading coefficient for the winter season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see HPXML Interior Shading) is used. - **Name:** ``window_interior_shading_winter`` - **Type:** ``Double`` @@ -1462,7 +1462,7 @@ Interior shading coefficient for the winter season, which if provided overrides **Windows: Summer Interior Shading Coefficient** -Interior shading coefficient for the summer season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see HPXML Interior Shading) is used. +Interior shading coefficient for the summer season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see HPXML Interior Shading) is used. - **Name:** ``window_interior_shading_summer`` - **Type:** ``Double`` @@ -1475,7 +1475,7 @@ Interior shading coefficient for the summer season, which if provided overrides **Windows: Exterior Shading Type** -Type of window exterior shading. Summer/winter shading coefficients can be provided below instead. If neither is provided, the OS-HPXML default (see HPXML Exterior Shading) is used. +Type of window exterior shading. Summer/winter shading coefficients can be provided below instead. If neither is provided, the OS-HPXML default (see HPXML Exterior Shading) is used. - **Name:** ``window_exterior_shading_type`` - **Type:** ``Choice`` @@ -1488,7 +1488,7 @@ Type of window exterior shading. Summer/winter shading coefficients can be provi **Windows: Winter Exterior Shading Coefficient** -Exterior shading coefficient for the winter season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see HPXML Exterior Shading) is used. +Exterior shading coefficient for the winter season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see HPXML Exterior Shading) is used. - **Name:** ``window_exterior_shading_winter`` - **Type:** ``Double`` @@ -1501,7 +1501,7 @@ Exterior shading coefficient for the winter season, which if provided overrides **Windows: Summer Exterior Shading Coefficient** -Exterior shading coefficient for the summer season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see HPXML Exterior Shading) is used. +Exterior shading coefficient for the summer season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see HPXML Exterior Shading) is used. - **Name:** ``window_exterior_shading_summer`` - **Type:** ``Double`` @@ -1514,7 +1514,7 @@ Exterior shading coefficient for the summer season, which if provided overrides **Windows: Shading Summer Season** -Enter a date range like 'May 1 - Sep 30'. Defines the summer season for purposes of shading coefficients; the rest of the year is assumed to be winter. If not provided, the OS-HPXML default (see HPXML Windows) is used. +Enter a date range like 'May 1 - Sep 30'. Defines the summer season for purposes of shading coefficients; the rest of the year is assumed to be winter. If not provided, the OS-HPXML default (see HPXML Windows) is used. - **Name:** ``window_shading_summer_season`` - **Type:** ``String`` @@ -1885,7 +1885,7 @@ Type of air leakage if providing a numeric air leakage value. If 'unit total', r **Air Leakage: Has Flue or Chimney in Conditioned Space** -Presence of flue or chimney with combustion air from conditioned space; used for infiltration model. If not provided, the OS-HPXML default (see Flue or Chimney) is used. +Presence of flue or chimney with combustion air from conditioned space; used for infiltration model. If not provided, the OS-HPXML default (see Flue or Chimney) is used. - **Name:** ``air_leakage_has_flue_or_chimney_in_conditioned_space`` - **Type:** ``Boolean`` @@ -1935,7 +1935,7 @@ The rated heating efficiency value of the heating system. **Heating System: Heating Capacity** -The output heating capacity of the heating system. If not provided, the OS-HPXML autosized default (see HPXML Heating Systems) is used. +The output heating capacity of the heating system. If not provided, the OS-HPXML autosized default (see HPXML Heating Systems) is used. - **Name:** ``heating_system_heating_capacity`` - **Type:** ``Double`` @@ -2048,7 +2048,7 @@ The rated efficiency value of the cooling system. Ignored for evaporative cooler **Cooling System: Cooling Compressor Type** -The compressor type of the cooling system. Only applies to central air conditioner and mini-split. If not provided, the OS-HPXML default (see Central Air Conditioner, Mini-Split Air Conditioner) is used. +The compressor type of the cooling system. Only applies to central air conditioner and mini-split. If not provided, the OS-HPXML default (see Central Air Conditioner, Mini-Split Air Conditioner) is used. - **Name:** ``cooling_system_cooling_compressor_type`` - **Type:** ``Choice`` @@ -2061,7 +2061,7 @@ The compressor type of the cooling system. Only applies to central air condition **Cooling System: Cooling Sensible Heat Fraction** -The sensible heat fraction of the cooling system. Ignored for evaporative cooler. If not provided, the OS-HPXML default (see Central Air Conditioner, Room Air Conditioner, Packaged Terminal Air Conditioner, Mini-Split Air Conditioner) is used. +The sensible heat fraction of the cooling system. Ignored for evaporative cooler. If not provided, the OS-HPXML default (see Central Air Conditioner, Room Air Conditioner, Packaged Terminal Air Conditioner, Mini-Split Air Conditioner) is used. - **Name:** ``cooling_system_cooling_sensible_heat_fraction`` - **Type:** ``Double`` @@ -2074,7 +2074,7 @@ The sensible heat fraction of the cooling system. Ignored for evaporative cooler **Cooling System: Cooling Capacity** -The output cooling capacity of the cooling system. If not provided, the OS-HPXML autosized default (see Central Air Conditioner, Room Air Conditioner, Packaged Terminal Air Conditioner, Evaporative Cooler, Mini-Split Air Conditioner) is used. +The output cooling capacity of the cooling system. If not provided, the OS-HPXML autosized default (see Central Air Conditioner, Room Air Conditioner, Packaged Terminal Air Conditioner, Evaporative Cooler, Mini-Split Air Conditioner) is used. - **Name:** ``cooling_system_cooling_capacity`` - **Type:** ``Double`` @@ -2161,7 +2161,7 @@ The refrigerant charge defect ratio, defined as (InstalledCharge - DesignCharge) **Cooling System: Crankcase Heater Power Watts** -Cooling system crankcase heater power consumption in Watts. Applies only to central air conditioner, room air conditioner, packaged terminal air conditioner and mini-split. If not provided, the OS-HPXML default (see Central Air Conditioner, Room Air Conditioner, Packaged Terminal Air Conditioner, Mini-Split Air Conditioner) is used. +Cooling system crankcase heater power consumption in Watts. Applies only to central air conditioner, room air conditioner, packaged terminal air conditioner and mini-split. If not provided, the OS-HPXML default (see Central Air Conditioner, Room Air Conditioner, Packaged Terminal Air Conditioner, Mini-Split Air Conditioner) is used. - **Name:** ``cooling_system_crankcase_heater_watts`` - **Type:** ``Double`` @@ -2200,7 +2200,7 @@ The rated heating efficiency value of the heating system integrated into cooling **Cooling System: Integrated Heating System Heating Capacity** -The output heating capacity of the heating system integrated into cooling system. If not provided, the OS-HPXML autosized default (see Room Air Conditioner, Packaged Terminal Air Conditioner) is used. Only used for room air conditioner and packaged terminal air conditioner. +The output heating capacity of the heating system integrated into cooling system. If not provided, the OS-HPXML autosized default (see Room Air Conditioner, Packaged Terminal Air Conditioner) is used. Only used for room air conditioner and packaged terminal air conditioner. - **Name:** ``cooling_system_integrated_heating_system_capacity`` - **Type:** ``Double`` @@ -2287,7 +2287,7 @@ The rated cooling efficiency value of the heat pump. **Heat Pump: Cooling Compressor Type** -The compressor type of the heat pump. Only applies to air-to-air and mini-split. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump) is used. +The compressor type of the heat pump. Only applies to air-to-air and mini-split. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump) is used. - **Name:** ``heat_pump_cooling_compressor_type`` - **Type:** ``Choice`` @@ -2300,7 +2300,7 @@ The compressor type of the heat pump. Only applies to air-to-air and mini-split. **Heat Pump: Cooling Sensible Heat Fraction** -The sensible heat fraction of the heat pump. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle, Ground-to-Air Heat Pump) is used. +The sensible heat fraction of the heat pump. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle, Ground-to-Air Heat Pump) is used. - **Name:** ``heat_pump_cooling_sensible_heat_fraction`` - **Type:** ``Double`` @@ -2313,7 +2313,7 @@ The sensible heat fraction of the heat pump. If not provided, the OS-HPXML defau **Heat Pump: Heating Capacity** -The output heating capacity of the heat pump. If not provided, the OS-HPXML autosized default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle, Ground-to-Air Heat Pump) is used. +The output heating capacity of the heat pump. If not provided, the OS-HPXML autosized default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle, Ground-to-Air Heat Pump) is used. - **Name:** ``heat_pump_heating_capacity`` - **Type:** ``Double`` @@ -2350,7 +2350,7 @@ The maximum capacity limit applied to the auto-sizing methodology. If not provid **Heat Pump: Heating Capacity Retention Fraction** -The output heating capacity of the heat pump at a user-specified temperature (e.g., 17F or 5F) divided by the above nominal heating capacity. Applies to all heat pump types except ground-to-air. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle) is used. +The output heating capacity of the heat pump at a user-specified temperature (e.g., 17F or 5F) divided by the above nominal heating capacity. Applies to all heat pump types except ground-to-air. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle) is used. - **Name:** ``heat_pump_heating_capacity_retention_fraction`` - **Type:** ``Double`` @@ -2376,7 +2376,7 @@ The user-specified temperature (e.g., 17F or 5F) for the above heating capacity **Heat Pump: Cooling Capacity** -The output cooling capacity of the heat pump. If not provided, the OS-HPXML autosized default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle, Ground-to-Air Heat Pump) is used. +The output cooling capacity of the heat pump. If not provided, the OS-HPXML autosized default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle, Ground-to-Air Heat Pump) is used. - **Name:** ``heat_pump_cooling_capacity`` - **Type:** ``Double`` @@ -2439,7 +2439,7 @@ The cooling load served by the heat pump. **Heat Pump: Compressor Lockout Temperature** -The temperature below which the heat pump compressor is disabled. If both this and Backup Heating Lockout Temperature are provided and use the same value, it essentially defines a switchover temperature (for, e.g., a dual-fuel heat pump). Applies to all heat pump types other than ground-to-air. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle) is used. +The temperature below which the heat pump compressor is disabled. If both this and Backup Heating Lockout Temperature are provided and use the same value, it essentially defines a switchover temperature (for, e.g., a dual-fuel heat pump). Applies to all heat pump types other than ground-to-air. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle) is used. - **Name:** ``heat_pump_compressor_lockout_temp`` - **Type:** ``Double`` @@ -2513,7 +2513,7 @@ The backup rated efficiency value of the heat pump. Percent for electricity fuel **Heat Pump: Backup Heating Capacity** -The backup output heating capacity of the heat pump. If not provided, the OS-HPXML autosized default (see Backup) is used. Only applies if Backup Type is 'integrated'. +The backup output heating capacity of the heat pump. If not provided, the OS-HPXML autosized default (see Backup) is used. Only applies if Backup Type is 'integrated'. - **Name:** ``heat_pump_backup_heating_capacity`` - **Type:** ``Double`` @@ -2526,7 +2526,7 @@ The backup output heating capacity of the heat pump. If not provided, the OS-HPX **Heat Pump: Backup Heating Lockout Temperature** -The temperature above which the heat pump backup system is disabled. If both this and Compressor Lockout Temperature are provided and use the same value, it essentially defines a switchover temperature (for, e.g., a dual-fuel heat pump). Applies for both Backup Type of 'integrated' and 'separate'. If not provided, the OS-HPXML default (see Backup) is used. +The temperature above which the heat pump backup system is disabled. If both this and Compressor Lockout Temperature are provided and use the same value, it essentially defines a switchover temperature (for, e.g., a dual-fuel heat pump). Applies for both Backup Type of 'integrated' and 'separate'. If not provided, the OS-HPXML default (see Backup) is used. - **Name:** ``heat_pump_backup_heating_lockout_temp`` - **Type:** ``Double`` @@ -2539,7 +2539,7 @@ The temperature above which the heat pump backup system is disabled. If both thi **Heat Pump: Sizing Methodology** -The auto-sizing methodology to use when the heat pump capacity is not provided. If not provided, the OS-HPXML default (see HPXML HVAC Sizing Control) is used. +The auto-sizing methodology to use when the heat pump capacity is not provided. If not provided, the OS-HPXML default (see HPXML HVAC Sizing Control) is used. - **Name:** ``heat_pump_sizing_methodology`` - **Type:** ``Choice`` @@ -2552,7 +2552,7 @@ The auto-sizing methodology to use when the heat pump capacity is not provided. **Heat Pump: Backup Sizing Methodology** -The auto-sizing methodology to use when the heat pump backup capacity is not provided. If not provided, the OS-HPXML default (see HPXML HVAC Sizing Control) is used. +The auto-sizing methodology to use when the heat pump backup capacity is not provided. If not provided, the OS-HPXML default (see HPXML HVAC Sizing Control) is used. - **Name:** ``heat_pump_backup_sizing_methodology`` - **Type:** ``Choice`` @@ -2602,7 +2602,7 @@ The refrigerant charge defect ratio, defined as (InstalledCharge - DesignCharge) **Heat Pump: Crankcase Heater Power Watts** -Heat Pump crankcase heater power consumption in Watts. Applies only to air-to-air, mini-split, packaged terminal heat pump and room air conditioner with reverse cycle. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle) is used. +Heat Pump crankcase heater power consumption in Watts. Applies only to air-to-air, mini-split, packaged terminal heat pump and room air conditioner with reverse cycle. If not provided, the OS-HPXML default (see Air-to-Air Heat Pump, Mini-Split Heat Pump, Packaged Terminal Heat Pump, Room Air Conditioner w/ Reverse Cycle) is used. - **Name:** ``heat_pump_crankcase_heater_watts`` - **Type:** ``Double`` @@ -2738,7 +2738,7 @@ Maximum speed efficiency COP values of cooling detailed performance data if avai **Geothermal Loop: Configuration** -Configuration of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see Ground-to-Air Heat Pump) is used. +Configuration of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see Ground-to-Air Heat Pump) is used. - **Name:** ``geothermal_loop_configuration`` - **Type:** ``Choice`` @@ -2751,7 +2751,7 @@ Configuration of the geothermal loop. Only applies to ground-to-air heat pump ty **Geothermal Loop: Borefield Configuration** -Borefield configuration of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. +Borefield configuration of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. - **Name:** ``geothermal_loop_borefield_configuration`` - **Type:** ``Choice`` @@ -2764,7 +2764,7 @@ Borefield configuration of the geothermal loop. Only applies to ground-to-air he **Geothermal Loop: Loop Flow** -Water flow rate through the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see HPXML Geothermal Loops) is used. +Water flow rate through the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see HPXML Geothermal Loops) is used. - **Name:** ``geothermal_loop_loop_flow`` - **Type:** ``Double`` @@ -2777,7 +2777,7 @@ Water flow rate through the geothermal loop. Only applies to ground-to-air heat **Geothermal Loop: Boreholes Count** -Number of boreholes. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see HPXML Geothermal Loops) is used. +Number of boreholes. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see HPXML Geothermal Loops) is used. - **Name:** ``geothermal_loop_boreholes_count`` - **Type:** ``Integer`` @@ -2790,7 +2790,7 @@ Number of boreholes. Only applies to ground-to-air heat pump type. If not provid **Geothermal Loop: Boreholes Length** -Average length of each borehole (vertical). Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see HPXML Geothermal Loops) is used. +Average length of each borehole (vertical). Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see HPXML Geothermal Loops) is used. - **Name:** ``geothermal_loop_boreholes_length`` - **Type:** ``Double`` @@ -2803,7 +2803,7 @@ Average length of each borehole (vertical). Only applies to ground-to-air heat p **Geothermal Loop: Boreholes Spacing** -Distance between bores. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. +Distance between bores. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. - **Name:** ``geothermal_loop_boreholes_spacing`` - **Type:** ``Double`` @@ -2816,7 +2816,7 @@ Distance between bores. Only applies to ground-to-air heat pump type. If not pro **Geothermal Loop: Boreholes Diameter** -Diameter of bores. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. +Diameter of bores. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. - **Name:** ``geothermal_loop_boreholes_diameter`` - **Type:** ``Double`` @@ -2829,7 +2829,7 @@ Diameter of bores. Only applies to ground-to-air heat pump type. If not provided **Geothermal Loop: Grout Type** -Grout type of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. +Grout type of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. - **Name:** ``geothermal_loop_grout_type`` - **Type:** ``Choice`` @@ -2842,7 +2842,7 @@ Grout type of the geothermal loop. Only applies to ground-to-air heat pump type. **Geothermal Loop: Pipe Type** -Pipe type of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. +Pipe type of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. - **Name:** ``geothermal_loop_pipe_type`` - **Type:** ``Choice`` @@ -2855,7 +2855,7 @@ Pipe type of the geothermal loop. Only applies to ground-to-air heat pump type. **Geothermal Loop: Pipe Diameter** -Pipe diameter of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. +Pipe diameter of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see HPXML Geothermal Loops) is used. - **Name:** ``geothermal_loop_pipe_diameter`` - **Type:** ``Choice`` @@ -2907,7 +2907,7 @@ The rated heating efficiency value of the second heating system. **Heating System 2: Heating Capacity** -The output heating capacity of the second heating system. If not provided, the OS-HPXML autosized default (see HPXML Heating Systems) is used. +The output heating capacity of the second heating system. If not provided, the OS-HPXML autosized default (see HPXML Heating Systems) is used. - **Name:** ``heating_system_2_heating_capacity`` - **Type:** ``Double`` @@ -3001,7 +3001,7 @@ Specify the constant or 24-hour comma-separated weekend cooling setpoint schedul **HVAC Control: Heating Season Period** -Enter a date range like 'Nov 1 - Jun 30'. If not provided, the OS-HPXML default (see HPXML HVAC Control) is used. Can also provide 'BuildingAmerica' to use automatic seasons from the Building America House Simulation Protocols. +Enter a date range like 'Nov 1 - Jun 30'. If not provided, the OS-HPXML default (see HPXML HVAC Control) is used. Can also provide 'BuildingAmerica' to use automatic seasons from the Building America House Simulation Protocols. - **Name:** ``hvac_control_heating_season_period`` - **Type:** ``String`` @@ -3012,7 +3012,7 @@ Enter a date range like 'Nov 1 - Jun 30'. If not provided, the OS-HPXML default **HVAC Control: Cooling Season Period** -Enter a date range like 'Jun 1 - Oct 31'. If not provided, the OS-HPXML default (see HPXML HVAC Control) is used. Can also provide 'BuildingAmerica' to use automatic seasons from the Building America House Simulation Protocols. +Enter a date range like 'Jun 1 - Oct 31'. If not provided, the OS-HPXML default (see HPXML HVAC Control) is used. Can also provide 'BuildingAmerica' to use automatic seasons from the Building America House Simulation Protocols. - **Name:** ``hvac_control_cooling_season_period`` - **Type:** ``String`` @@ -3023,7 +3023,7 @@ Enter a date range like 'Jun 1 - Oct 31'. If not provided, the OS-HPXML default **HVAC Blower: Fan Efficiency** -The blower fan efficiency at maximum fan speed. Applies only to split (not packaged) systems (i.e., applies to ducted systems as well as ductless mini-split systems). If not provided, the OS-HPXML default (see HPXML Heating Systems, HPXML Cooling Systems, HPXML Heat Pumps) is used. +The blower fan efficiency at maximum fan speed. Applies only to split (not packaged) systems (i.e., applies to ducted systems as well as ductless mini-split systems). If not provided, the OS-HPXML default (see HPXML Heating Systems, HPXML Cooling Systems, HPXML Heat Pumps) is used. - **Name:** ``hvac_blower_fan_watts_per_cfm`` - **Type:** ``Double`` @@ -3060,7 +3060,7 @@ The leakage value to outside for the supply ducts. **Ducts: Supply Location** -The location of the supply ducts. If not provided, the OS-HPXML default (see Air Distribution) is used. +The location of the supply ducts. If not provided, the OS-HPXML default (see Air Distribution) is used. - **Name:** ``ducts_supply_location`` - **Type:** ``Choice`` @@ -3099,7 +3099,7 @@ Whether the supply ducts are buried in, e.g., attic loose-fill insulation. Parti **Ducts: Supply Surface Area** -The supply ducts surface area in the given location. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see Air Distribution) is used. +The supply ducts surface area in the given location. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see Air Distribution) is used. - **Name:** ``ducts_supply_surface_area`` - **Type:** ``Double`` @@ -3112,7 +3112,7 @@ The supply ducts surface area in the given location. If neither Surface Area nor **Ducts: Supply Area Fraction** -The fraction of supply ducts surface area in the given location. Only used if Surface Area is not provided. If the fraction is less than 1, the remaining duct area is assumed to be in conditioned space. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see Air Distribution) is used. +The fraction of supply ducts surface area in the given location. Only used if Surface Area is not provided. If the fraction is less than 1, the remaining duct area is assumed to be in conditioned space. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see Air Distribution) is used. - **Name:** ``ducts_supply_surface_area_fraction`` - **Type:** ``Double`` @@ -3125,7 +3125,7 @@ The fraction of supply ducts surface area in the given location. Only used if Su **Ducts: Supply Fraction Rectangular** -The fraction of supply ducts that are rectangular (as opposed to round); this affects the duct effective R-value used for modeling. If not provided, the OS-HPXML default (see Air Distribution) is used. +The fraction of supply ducts that are rectangular (as opposed to round); this affects the duct effective R-value used for modeling. If not provided, the OS-HPXML default (see Air Distribution) is used. - **Name:** ``ducts_supply_fraction_rectangular`` - **Type:** ``Double`` @@ -3149,7 +3149,7 @@ The leakage value to outside for the return ducts. **Ducts: Return Location** -The location of the return ducts. If not provided, the OS-HPXML default (see Air Distribution) is used. +The location of the return ducts. If not provided, the OS-HPXML default (see Air Distribution) is used. - **Name:** ``ducts_return_location`` - **Type:** ``Choice`` @@ -3188,7 +3188,7 @@ Whether the return ducts are buried in, e.g., attic loose-fill insulation. Parti **Ducts: Return Surface Area** -The return ducts surface area in the given location. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see Air Distribution) is used. +The return ducts surface area in the given location. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see Air Distribution) is used. - **Name:** ``ducts_return_surface_area`` - **Type:** ``Double`` @@ -3201,7 +3201,7 @@ The return ducts surface area in the given location. If neither Surface Area nor **Ducts: Return Area Fraction** -The fraction of return ducts surface area in the given location. Only used if Surface Area is not provided. If the fraction is less than 1, the remaining duct area is assumed to be in conditioned space. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see Air Distribution) is used. +The fraction of return ducts surface area in the given location. Only used if Surface Area is not provided. If the fraction is less than 1, the remaining duct area is assumed to be in conditioned space. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see Air Distribution) is used. - **Name:** ``ducts_return_surface_area_fraction`` - **Type:** ``Double`` @@ -3214,7 +3214,7 @@ The fraction of return ducts surface area in the given location. Only used if Su **Ducts: Number of Return Registers** -The number of return registers of the ducts. Only used to calculate default return duct surface area. If not provided, the OS-HPXML default (see Air Distribution) is used. +The number of return registers of the ducts. Only used to calculate default return duct surface area. If not provided, the OS-HPXML default (see Air Distribution) is used. - **Name:** ``ducts_number_of_return_registers`` - **Type:** ``Integer`` @@ -3227,7 +3227,7 @@ The number of return registers of the ducts. Only used to calculate default retu **Ducts: Return Fraction Rectangular** -The fraction of return ducts that are rectangular (as opposed to round); this affects the duct effective R-value used for modeling. If not provided, the OS-HPXML default (see Air Distribution) is used. +The fraction of return ducts that are rectangular (as opposed to round); this affects the duct effective R-value used for modeling. If not provided, the OS-HPXML default (see Air Distribution) is used. - **Name:** ``ducts_return_fraction_rectangular`` - **Type:** ``Double`` @@ -3253,7 +3253,7 @@ The type of the mechanical ventilation. Use 'none' if there is no mechanical ven **Mechanical Ventilation: Flow Rate** -The flow rate of the mechanical ventilation. If not provided, the OS-HPXML default (see HPXML Mechanical Ventilation Fans) is used. +The flow rate of the mechanical ventilation. If not provided, the OS-HPXML default (see HPXML Mechanical Ventilation Fans) is used. - **Name:** ``mech_vent_flow_rate`` - **Type:** ``Double`` @@ -3266,7 +3266,7 @@ The flow rate of the mechanical ventilation. If not provided, the OS-HPXML defau **Mechanical Ventilation: Hours In Operation** -The hours in operation of the mechanical ventilation. If not provided, the OS-HPXML default (see HPXML Mechanical Ventilation Fans) is used. +The hours in operation of the mechanical ventilation. If not provided, the OS-HPXML default (see HPXML Mechanical Ventilation Fans) is used. - **Name:** ``mech_vent_hours_in_operation`` - **Type:** ``Double`` @@ -3318,7 +3318,7 @@ The Unadjusted or Adjusted sensible recovery efficiency of the mechanical ventil **Mechanical Ventilation: Fan Power** -The fan power of the mechanical ventilation. If not provided, the OS-HPXML default (see HPXML Mechanical Ventilation Fans) is used. +The fan power of the mechanical ventilation. If not provided, the OS-HPXML default (see HPXML Mechanical Ventilation Fans) is used. - **Name:** ``mech_vent_fan_power`` - **Type:** ``Double`` @@ -3526,7 +3526,7 @@ The fan power of the second mechanical ventilation. **Kitchen Fans: Quantity** -The quantity of the kitchen fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The quantity of the kitchen fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``kitchen_fans_quantity`` - **Type:** ``Integer`` @@ -3539,7 +3539,7 @@ The quantity of the kitchen fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The flow rate of the kitchen fan. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``kitchen_fans_flow_rate`` - **Type:** ``Double`` @@ -3552,7 +3552,7 @@ The flow rate of the kitchen fan. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The hours in operation of the kitchen fan. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``kitchen_fans_hours_in_operation`` - **Type:** ``Double`` @@ -3565,7 +3565,7 @@ The hours in operation of the kitchen fan. If not provided, the OS-HPXML default **Kitchen Fans: Fan Power** -The fan power of the kitchen fan. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The fan power of the kitchen fan. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``kitchen_fans_power`` - **Type:** ``Double`` @@ -3578,7 +3578,7 @@ The fan power of the kitchen fan. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The start hour of the kitchen fan. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``kitchen_fans_start_hour`` - **Type:** ``Integer`` @@ -3591,7 +3591,7 @@ The start hour of the kitchen fan. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The quantity of the bathroom fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``bathroom_fans_quantity`` - **Type:** ``Integer`` @@ -3604,7 +3604,7 @@ The quantity of the bathroom fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The flow rate of the bathroom fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``bathroom_fans_flow_rate`` - **Type:** ``Double`` @@ -3617,7 +3617,7 @@ The flow rate of the bathroom fans. If not provided, the OS-HPXML default (see < **Bathroom Fans: Hours In Operation** -The hours in operation of the bathroom fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The hours in operation of the bathroom fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``bathroom_fans_hours_in_operation`` - **Type:** ``Double`` @@ -3630,7 +3630,7 @@ The hours in operation of the bathroom fans. If not provided, the OS-HPXML defau **Bathroom Fans: Fan Power** -The fan power of the bathroom fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The fan power of the bathroom fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``bathroom_fans_power`` - **Type:** ``Double`` @@ -3643,7 +3643,7 @@ The fan power of the bathroom fans. If not provided, the OS-HPXML default (see < **Bathroom Fans: Start Hour** -The start hour of the bathroom fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. +The start hour of the bathroom fans. If not provided, the OS-HPXML default (see HPXML Local Ventilation Fans) is used. - **Name:** ``bathroom_fans_start_hour`` - **Type:** ``Integer`` @@ -3667,7 +3667,7 @@ Whether there is a whole house fan. **Whole House Fan: Flow Rate** -The flow rate of the whole house fan. If not provided, the OS-HPXML default (see HPXML Whole House Fans) is used. +The flow rate of the whole house fan. If not provided, the OS-HPXML default (see HPXML Whole House Fans) is used. - **Name:** ``whole_house_fan_flow_rate`` - **Type:** ``Double`` @@ -3680,7 +3680,7 @@ The flow rate of the whole house fan. If not provided, the OS-HPXML default (see **Whole House Fan: Fan Power** -The fan power of the whole house fan. If not provided, the OS-HPXML default (see HPXML Whole House Fans) is used. +The fan power of the whole house fan. If not provided, the OS-HPXML default (see HPXML Whole House Fans) is used. - **Name:** ``whole_house_fan_power`` - **Type:** ``Double`` @@ -3719,7 +3719,7 @@ The fuel type of water heater. Ignored for heat pump water heater. **Water Heater: Location** -The location of water heater. If not provided, the OS-HPXML default (see HPXML Water Heating Systems) is used. +The location of water heater. If not provided, the OS-HPXML default (see HPXML Water Heating Systems) is used. - **Name:** ``water_heater_location`` - **Type:** ``Choice`` @@ -3732,7 +3732,7 @@ The location of water heater. If not provided, the OS-HPXML default (see Conventional Storage, Heat Pump, Combi Boiler w/ Storage) is used. +Nominal volume of water heater tank. If not provided, the OS-HPXML default (see Conventional Storage, Heat Pump, Combi Boiler w/ Storage) is used. - **Name:** ``water_heater_tank_volume`` - **Type:** ``Double`` @@ -3769,7 +3769,7 @@ Rated Energy Factor or Uniform Energy Factor. Does not apply to space-heating bo **Water Heater: Usage Bin** -The usage of the water heater. Only applies if Efficiency Type is UniformEnergyFactor and Type is not instantaneous water heater. Does not apply to space-heating boilers. If not provided, the OS-HPXML default (see Conventional Storage, Heat Pump) is used. +The usage of the water heater. Only applies if Efficiency Type is UniformEnergyFactor and Type is not instantaneous water heater. Does not apply to space-heating boilers. If not provided, the OS-HPXML default (see Conventional Storage, Heat Pump) is used. - **Name:** ``water_heater_usage_bin`` - **Type:** ``Choice`` @@ -3782,7 +3782,7 @@ The usage of the water heater. Only applies if Efficiency Type is UniformEnergyF **Water Heater: Recovery Efficiency** -Ratio of energy delivered to water heater to the energy content of the fuel consumed by the water heater. Only used for non-electric storage water heaters. If not provided, the OS-HPXML default (see Conventional Storage) is used. +Ratio of energy delivered to water heater to the energy content of the fuel consumed by the water heater. Only used for non-electric storage water heaters. If not provided, the OS-HPXML default (see Conventional Storage) is used. - **Name:** ``water_heater_recovery_efficiency`` - **Type:** ``Double`` @@ -3795,7 +3795,7 @@ Ratio of energy delivered to water heater to the energy content of the fuel cons **Water Heater: Heating Capacity** -Heating capacity. Only applies to storage water heater and heat pump water heater (compressor). If not provided, the OS-HPXML default (see Conventional Storage, Heat Pump) is used. +Heating capacity. Only applies to storage water heater and heat pump water heater (compressor). If not provided, the OS-HPXML default (see Conventional Storage, Heat Pump) is used. - **Name:** ``water_heater_heating_capacity`` - **Type:** ``Double`` @@ -3808,7 +3808,7 @@ Heating capacity. Only applies to storage water heater and heat pump water heate **Water Heater: Backup Heating Capacity** -Backup heating capacity for a heat pump water heater. If not provided, the OS-HPXML default (see Heat Pump) is used. +Backup heating capacity for a heat pump water heater. If not provided, the OS-HPXML default (see Heat Pump) is used. - **Name:** ``water_heater_backup_heating_capacity`` - **Type:** ``Double`` @@ -3821,7 +3821,7 @@ Backup heating capacity for a heat pump water heater. If not provided, the OS-HP **Water Heater: Standby Loss** -The standby loss of water heater. Only applies to space-heating boilers. If not provided, the OS-HPXML default (see Combi Boiler w/ Storage) is used. +The standby loss of water heater. Only applies to space-heating boilers. If not provided, the OS-HPXML default (see Combi Boiler w/ Storage) is used. - **Name:** ``water_heater_standby_loss`` - **Type:** ``Double`` @@ -3847,7 +3847,7 @@ The jacket R-value of water heater. Doesn't apply to instantaneous water heater **Water Heater: Setpoint Temperature** -The setpoint temperature of water heater. If not provided, the OS-HPXML default (see HPXML Water Heating Systems) is used. +The setpoint temperature of water heater. If not provided, the OS-HPXML default (see HPXML Water Heating Systems) is used. - **Name:** ``water_heater_setpoint_temperature`` - **Type:** ``Double`` @@ -3884,7 +3884,7 @@ Requires that the dwelling unit has a air-to-air, mini-split, or ground-to-air h **Water Heater: Tank Type** -Type of tank model to use. The 'stratified' tank generally provide more accurate results, but may significantly increase run time. Applies only to storage water heater. If not provided, the OS-HPXML default (see Conventional Storage) is used. +Type of tank model to use. The 'stratified' tank generally provide more accurate results, but may significantly increase run time. Applies only to storage water heater. If not provided, the OS-HPXML default (see Conventional Storage) is used. - **Name:** ``water_heater_tank_model_type`` - **Type:** ``Choice`` @@ -3897,7 +3897,7 @@ Type of tank model to use. The 'stratified' tank generally provide more accurate **Water Heater: Operating Mode** -The water heater operating mode. The 'heat pump only' option only uses the heat pump, while 'hybrid/auto' allows the backup electric resistance to come on in high demand situations. This is ignored if a scheduled operating mode type is selected. Applies only to heat pump water heater. If not provided, the OS-HPXML default (see Heat Pump) is used. +The water heater operating mode. The 'heat pump only' option only uses the heat pump, while 'hybrid/auto' allows the backup electric resistance to come on in high demand situations. This is ignored if a scheduled operating mode type is selected. Applies only to heat pump water heater. If not provided, the OS-HPXML default (see Heat Pump) is used. - **Name:** ``water_heater_operating_mode`` - **Type:** ``Choice`` @@ -3923,7 +3923,7 @@ The type of the hot water distribution system. **Hot Water Distribution: Standard Piping Length** -If the distribution system is Standard, the length of the piping. If not provided, the OS-HPXML default (see Standard) is used. +If the distribution system is Standard, the length of the piping. If not provided, the OS-HPXML default (see Standard) is used. - **Name:** ``hot_water_distribution_standard_piping_length`` - **Type:** ``Double`` @@ -3949,7 +3949,7 @@ If the distribution system is Recirculation, the type of hot water recirculation **Hot Water Distribution: Recirculation Piping Length** -If the distribution system is Recirculation, the length of the recirculation piping. If not provided, the OS-HPXML default (see Recirculation (In-Unit)) is used. +If the distribution system is Recirculation, the length of the recirculation piping. If not provided, the OS-HPXML default (see Recirculation (In-Unit)) is used. - **Name:** ``hot_water_distribution_recirc_piping_length`` - **Type:** ``Double`` @@ -3962,7 +3962,7 @@ If the distribution system is Recirculation, the length of the recirculation pip **Hot Water Distribution: Recirculation Branch Piping Length** -If the distribution system is Recirculation, the length of the recirculation branch piping. If not provided, the OS-HPXML default (see Recirculation (In-Unit)) is used. +If the distribution system is Recirculation, the length of the recirculation branch piping. If not provided, the OS-HPXML default (see Recirculation (In-Unit)) is used. - **Name:** ``hot_water_distribution_recirc_branch_piping_length`` - **Type:** ``Double`` @@ -3975,7 +3975,7 @@ If the distribution system is Recirculation, the length of the recirculation bra **Hot Water Distribution: Recirculation Pump Power** -If the distribution system is Recirculation, the recirculation pump power. If not provided, the OS-HPXML default (see Recirculation (In-Unit)) is used. +If the distribution system is Recirculation, the recirculation pump power. If not provided, the OS-HPXML default (see Recirculation (In-Unit)) is used. - **Name:** ``hot_water_distribution_recirc_pump_power`` - **Type:** ``Double`` @@ -3988,7 +3988,7 @@ If the distribution system is Recirculation, the recirculation pump power. If no **Hot Water Distribution: Pipe Insulation Nominal R-Value** -Nominal R-value of the pipe insulation. If not provided, the OS-HPXML default (see HPXML Hot Water Distribution) is used. +Nominal R-value of the pipe insulation. If not provided, the OS-HPXML default (see HPXML Hot Water Distribution) is used. - **Name:** ``hot_water_distribution_pipe_r`` - **Type:** ``Double`` @@ -4060,7 +4060,7 @@ Whether the sink fixture is low flow. **Hot Water Fixtures: Usage Multiplier** -Multiplier on the hot water usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Water Fixtures) is used. +Multiplier on the hot water usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Water Fixtures) is used. - **Name:** ``water_fixtures_usage_multiplier`` - **Type:** ``Double`` @@ -4071,7 +4071,7 @@ Multiplier on the hot water usage that can reflect, e.g., high/low usage occupan **General Water Use: Usage Multiplier** -Multiplier on internal gains from general water use (floor mopping, shower evaporation, water films on showers, tubs & sinks surfaces, plant watering, etc.) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Building Occupancy) is used. +Multiplier on internal gains from general water use (floor mopping, shower evaporation, water films on showers, tubs & sinks surfaces, plant watering, etc.) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Building Occupancy) is used. - **Name:** ``general_water_use_usage_multiplier`` - **Type:** ``Double`` @@ -4184,7 +4184,7 @@ The collector rated thermal losses of the solar thermal system. **Solar Thermal: Storage Volume** -The storage volume of the solar thermal system. If not provided, the OS-HPXML default (see Detailed Inputs) is used. +The storage volume of the solar thermal system. If not provided, the OS-HPXML default (see Detailed Inputs) is used. - **Name:** ``solar_thermal_storage_volume`` - **Type:** ``Double`` @@ -4221,7 +4221,7 @@ Whether there is a PV system present. **PV System: Module Type** -Module type of the PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. +Module type of the PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. - **Name:** ``pv_system_module_type`` - **Type:** ``Choice`` @@ -4234,7 +4234,7 @@ Module type of the PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. +Location of the PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. - **Name:** ``pv_system_location`` - **Type:** ``Choice`` @@ -4247,7 +4247,7 @@ Location of the PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. +Type of tracking for the PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. - **Name:** ``pv_system_tracking`` - **Type:** ``Choice`` @@ -4297,7 +4297,7 @@ Maximum power output of the PV system. For a shared system, this is the total bu **PV System: Inverter Efficiency** -Inverter efficiency of the PV system. If there are two PV systems, this will apply to both. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. +Inverter efficiency of the PV system. If there are two PV systems, this will apply to both. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. - **Name:** ``pv_system_inverter_efficiency`` - **Type:** ``Double`` @@ -4310,7 +4310,7 @@ Inverter efficiency of the PV system. If there are two PV systems, this will app **PV System: System Losses Fraction** -System losses fraction of the PV system. If there are two PV systems, this will apply to both. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. +System losses fraction of the PV system. If there are two PV systems, this will apply to both. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. - **Name:** ``pv_system_system_losses_fraction`` - **Type:** ``Double`` @@ -4347,7 +4347,7 @@ Whether there is a second PV system present. **PV System 2: Module Type** -Module type of the second PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. +Module type of the second PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. - **Name:** ``pv_system_2_module_type`` - **Type:** ``Choice`` @@ -4360,7 +4360,7 @@ Module type of the second PV system. If not provided, the OS-HPXML default (see **PV System 2: Location** -Location of the second PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. +Location of the second PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. - **Name:** ``pv_system_2_location`` - **Type:** ``Choice`` @@ -4373,7 +4373,7 @@ Location of the second PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. +Type of tracking for the second PV system. If not provided, the OS-HPXML default (see HPXML Photovoltaics) is used. - **Name:** ``pv_system_2_tracking`` - **Type:** ``Choice`` @@ -4434,7 +4434,7 @@ Whether there is a lithium ion battery present. **Battery: Location** -The space type for the lithium ion battery location. If not provided, the OS-HPXML default (see HPXML Batteries) is used. +The space type for the lithium ion battery location. If not provided, the OS-HPXML default (see HPXML Batteries) is used. - **Name:** ``battery_location`` - **Type:** ``Choice`` @@ -4447,7 +4447,7 @@ The space type for the lithium ion battery location. If not provided, the OS-HPX **Battery: Rated Power Output** -The rated power output of the lithium ion battery. If not provided, the OS-HPXML default (see HPXML Batteries) is used. +The rated power output of the lithium ion battery. If not provided, the OS-HPXML default (see HPXML Batteries) is used. - **Name:** ``battery_power`` - **Type:** ``Double`` @@ -4460,7 +4460,7 @@ The rated power output of the lithium ion battery. If not provided, the OS-HPXML **Battery: Nominal Capacity** -The nominal capacity of the lithium ion battery. If not provided, the OS-HPXML default (see HPXML Batteries) is used. +The nominal capacity of the lithium ion battery. If not provided, the OS-HPXML default (see HPXML Batteries) is used. - **Name:** ``battery_capacity`` - **Type:** ``Double`` @@ -4473,7 +4473,7 @@ The nominal capacity of the lithium ion battery. If not provided, the OS-HPXML d **Battery: Usable Capacity** -The usable capacity of the lithium ion battery. If not provided, the OS-HPXML default (see HPXML Batteries) is used. +The usable capacity of the lithium ion battery. If not provided, the OS-HPXML default (see HPXML Batteries) is used. - **Name:** ``battery_usable_capacity`` - **Type:** ``Double`` @@ -4486,7 +4486,7 @@ The usable capacity of the lithium ion battery. If not provided, the OS-HPXML de **Battery: Round Trip Efficiency** -The round trip efficiency of the lithium ion battery. If not provided, the OS-HPXML default (see HPXML Batteries) is used. +The round trip efficiency of the lithium ion battery. If not provided, the OS-HPXML default (see HPXML Batteries) is used. - **Name:** ``battery_round_trip_efficiency`` - **Type:** ``Double`` @@ -4556,7 +4556,7 @@ Fraction of all lamps (interior) that are light emitting diodes. Lighting not sp **Lighting: Interior Usage Multiplier** -Multiplier on the lighting energy usage (interior) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Lighting) is used. +Multiplier on the lighting energy usage (interior) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Lighting) is used. - **Name:** ``lighting_interior_usage_multiplier`` - **Type:** ``Double`` @@ -4600,7 +4600,7 @@ Fraction of all lamps (exterior) that are light emitting diodes. Lighting not sp **Lighting: Exterior Usage Multiplier** -Multiplier on the lighting energy usage (exterior) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Lighting) is used. +Multiplier on the lighting energy usage (exterior) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Lighting) is used. - **Name:** ``lighting_exterior_usage_multiplier`` - **Type:** ``Double`` @@ -4644,7 +4644,7 @@ Fraction of all lamps (garage) that are light emitting diodes. Lighting not spec **Lighting: Garage Usage Multiplier** -Multiplier on the lighting energy usage (garage) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Lighting) is used. +Multiplier on the lighting energy usage (garage) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Lighting) is used. - **Name:** ``lighting_garage_usage_multiplier`` - **Type:** ``Double`` @@ -4666,7 +4666,7 @@ Whether there is holiday lighting. **Holiday Lighting: Daily Consumption** -The daily energy consumption for holiday lighting (exterior). If not provided, the OS-HPXML default (see HPXML Lighting) is used. +The daily energy consumption for holiday lighting (exterior). If not provided, the OS-HPXML default (see HPXML Lighting) is used. - **Name:** ``holiday_lighting_daily_kwh`` - **Type:** ``Double`` @@ -4679,7 +4679,7 @@ The daily energy consumption for holiday lighting (exterior). If not provided, t **Holiday Lighting: Period** -Enter a date range like 'Nov 25 - Jan 5'. If not provided, the OS-HPXML default (see HPXML Lighting) is used. +Enter a date range like 'Nov 25 - Jan 5'. If not provided, the OS-HPXML default (see HPXML Lighting) is used. - **Name:** ``holiday_lighting_period`` - **Type:** ``String`` @@ -4779,7 +4779,7 @@ Whether there is a clothes washer present. **Clothes Washer: Location** -The space type for the clothes washer location. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. +The space type for the clothes washer location. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. - **Name:** ``clothes_washer_location`` - **Type:** ``Choice`` @@ -4805,7 +4805,7 @@ The efficiency type of the clothes washer. **Clothes Washer: Efficiency** -The efficiency of the clothes washer. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. +The efficiency of the clothes washer. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. - **Name:** ``clothes_washer_efficiency`` - **Type:** ``Double`` @@ -4818,7 +4818,7 @@ The efficiency of the clothes washer. If not provided, the OS-HPXML default (see **Clothes Washer: Rated Annual Consumption** -The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. +The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. - **Name:** ``clothes_washer_rated_annual_kwh`` - **Type:** ``Double`` @@ -4831,7 +4831,7 @@ The annual energy consumed by the clothes washer, as rated, obtained from the En **Clothes Washer: Label Electric Rate** -The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. +The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. - **Name:** ``clothes_washer_label_electric_rate`` - **Type:** ``Double`` @@ -4844,7 +4844,7 @@ The annual energy consumed by the clothes washer, as rated, obtained from the En **Clothes Washer: Label Gas Rate** -The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. +The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. - **Name:** ``clothes_washer_label_gas_rate`` - **Type:** ``Double`` @@ -4857,7 +4857,7 @@ The annual energy consumed by the clothes washer, as rated, obtained from the En **Clothes Washer: Label Annual Cost with Gas DHW** -The annual cost of using the system under test conditions. Input is obtained from the EnergyGuide label. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. +The annual cost of using the system under test conditions. Input is obtained from the EnergyGuide label. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. - **Name:** ``clothes_washer_label_annual_gas_cost`` - **Type:** ``Double`` @@ -4870,7 +4870,7 @@ The annual cost of using the system under test conditions. Input is obtained fro **Clothes Washer: Label Usage** -The clothes washer loads per week. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. +The clothes washer loads per week. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. - **Name:** ``clothes_washer_label_usage`` - **Type:** ``Double`` @@ -4883,7 +4883,7 @@ The clothes washer loads per week. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. +Volume of the washer drum. Obtained from the EnergyStar website or the manufacturer's literature. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. - **Name:** ``clothes_washer_capacity`` - **Type:** ``Double`` @@ -4896,7 +4896,7 @@ Volume of the washer drum. Obtained from the EnergyStar website or the manufactu **Clothes Washer: Usage Multiplier** -Multiplier on the clothes washer energy and hot water usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. +Multiplier on the clothes washer energy and hot water usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Clothes Washer) is used. - **Name:** ``clothes_washer_usage_multiplier`` - **Type:** ``Double`` @@ -4918,7 +4918,7 @@ Whether there is a clothes dryer present. **Clothes Dryer: Location** -The space type for the clothes dryer location. If not provided, the OS-HPXML default (see HPXML Clothes Dryer) is used. +The space type for the clothes dryer location. If not provided, the OS-HPXML default (see HPXML Clothes Dryer) is used. - **Name:** ``clothes_dryer_location`` - **Type:** ``Choice`` @@ -4957,7 +4957,7 @@ The efficiency type of the clothes dryer. **Clothes Dryer: Efficiency** -The efficiency of the clothes dryer. If not provided, the OS-HPXML default (see HPXML Clothes Dryer) is used. +The efficiency of the clothes dryer. If not provided, the OS-HPXML default (see HPXML Clothes Dryer) is used. - **Name:** ``clothes_dryer_efficiency`` - **Type:** ``Double`` @@ -4970,7 +4970,7 @@ The efficiency of the clothes dryer. If not provided, the OS-HPXML default (see **Clothes Dryer: Vented Flow Rate** -The exhaust flow rate of the vented clothes dryer. If not provided, the OS-HPXML default (see HPXML Clothes Dryer) is used. +The exhaust flow rate of the vented clothes dryer. If not provided, the OS-HPXML default (see HPXML Clothes Dryer) is used. - **Name:** ``clothes_dryer_vented_flow_rate`` - **Type:** ``Double`` @@ -4983,7 +4983,7 @@ The exhaust flow rate of the vented clothes dryer. If not provided, the OS-HPXML **Clothes Dryer: Usage Multiplier** -Multiplier on the clothes dryer energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Clothes Dryer) is used. +Multiplier on the clothes dryer energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Clothes Dryer) is used. - **Name:** ``clothes_dryer_usage_multiplier`` - **Type:** ``Double`` @@ -5005,7 +5005,7 @@ Whether there is a dishwasher present. **Dishwasher: Location** -The space type for the dishwasher location. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. +The space type for the dishwasher location. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. - **Name:** ``dishwasher_location`` - **Type:** ``Choice`` @@ -5031,7 +5031,7 @@ The efficiency type of dishwasher. **Dishwasher: Efficiency** -The efficiency of the dishwasher. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. +The efficiency of the dishwasher. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. - **Name:** ``dishwasher_efficiency`` - **Type:** ``Double`` @@ -5044,7 +5044,7 @@ The efficiency of the dishwasher. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. +The label electric rate of the dishwasher. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. - **Name:** ``dishwasher_label_electric_rate`` - **Type:** ``Double`` @@ -5057,7 +5057,7 @@ The label electric rate of the dishwasher. If not provided, the OS-HPXML default **Dishwasher: Label Gas Rate** -The label gas rate of the dishwasher. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. +The label gas rate of the dishwasher. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. - **Name:** ``dishwasher_label_gas_rate`` - **Type:** ``Double`` @@ -5070,7 +5070,7 @@ The label gas rate of the dishwasher. If not provided, the OS-HPXML default (see **Dishwasher: Label Annual Gas Cost** -The label annual gas cost of the dishwasher. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. +The label annual gas cost of the dishwasher. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. - **Name:** ``dishwasher_label_annual_gas_cost`` - **Type:** ``Double`` @@ -5083,7 +5083,7 @@ The label annual gas cost of the dishwasher. If not provided, the OS-HPXML defau **Dishwasher: Label Usage** -The dishwasher loads per week. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. +The dishwasher loads per week. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. - **Name:** ``dishwasher_label_usage`` - **Type:** ``Double`` @@ -5096,7 +5096,7 @@ The dishwasher loads per week. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. +The number of place settings for the unit. Data obtained from manufacturer's literature. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. - **Name:** ``dishwasher_place_setting_capacity`` - **Type:** ``Integer`` @@ -5109,7 +5109,7 @@ The number of place settings for the unit. Data obtained from manufacturer's lit **Dishwasher: Usage Multiplier** -Multiplier on the dishwasher energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. +Multiplier on the dishwasher energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Dishwasher) is used. - **Name:** ``dishwasher_usage_multiplier`` - **Type:** ``Double`` @@ -5131,7 +5131,7 @@ Whether there is a refrigerator present. **Refrigerator: Location** -The space type for the refrigerator location. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. +The space type for the refrigerator location. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. - **Name:** ``refrigerator_location`` - **Type:** ``Choice`` @@ -5144,7 +5144,7 @@ The space type for the refrigerator location. If not provided, the OS-HPXML defa **Refrigerator: Rated Annual Consumption** -The EnergyGuide rated annual energy consumption for a refrigerator. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. +The EnergyGuide rated annual energy consumption for a refrigerator. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. - **Name:** ``refrigerator_rated_annual_kwh`` - **Type:** ``Double`` @@ -5157,7 +5157,7 @@ The EnergyGuide rated annual energy consumption for a refrigerator. If not provi **Refrigerator: Usage Multiplier** -Multiplier on the refrigerator energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. +Multiplier on the refrigerator energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. - **Name:** ``refrigerator_usage_multiplier`` - **Type:** ``Double`` @@ -5179,7 +5179,7 @@ Whether there is an extra refrigerator present. **Extra Refrigerator: Location** -The space type for the extra refrigerator location. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. +The space type for the extra refrigerator location. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. - **Name:** ``extra_refrigerator_location`` - **Type:** ``Choice`` @@ -5192,7 +5192,7 @@ The space type for the extra refrigerator location. If not provided, the OS-HPXM **Extra Refrigerator: Rated Annual Consumption** -The EnergyGuide rated annual energy consumption for an extra refrigerator. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. +The EnergyGuide rated annual energy consumption for an extra refrigerator. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. - **Name:** ``extra_refrigerator_rated_annual_kwh`` - **Type:** ``Double`` @@ -5205,7 +5205,7 @@ The EnergyGuide rated annual energy consumption for an extra refrigerator. If no **Extra Refrigerator: Usage Multiplier** -Multiplier on the extra refrigerator energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. +Multiplier on the extra refrigerator energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Refrigerators) is used. - **Name:** ``extra_refrigerator_usage_multiplier`` - **Type:** ``Double`` @@ -5227,7 +5227,7 @@ Whether there is a freezer present. **Freezer: Location** -The space type for the freezer location. If not provided, the OS-HPXML default (see HPXML Freezers) is used. +The space type for the freezer location. If not provided, the OS-HPXML default (see HPXML Freezers) is used. - **Name:** ``freezer_location`` - **Type:** ``Choice`` @@ -5240,7 +5240,7 @@ The space type for the freezer location. If not provided, the OS-HPXML default ( **Freezer: Rated Annual Consumption** -The EnergyGuide rated annual energy consumption for a freezer. If not provided, the OS-HPXML default (see HPXML Freezers) is used. +The EnergyGuide rated annual energy consumption for a freezer. If not provided, the OS-HPXML default (see HPXML Freezers) is used. - **Name:** ``freezer_rated_annual_kwh`` - **Type:** ``Double`` @@ -5253,7 +5253,7 @@ The EnergyGuide rated annual energy consumption for a freezer. If not provided, **Freezer: Usage Multiplier** -Multiplier on the freezer energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Freezers) is used. +Multiplier on the freezer energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Freezers) is used. - **Name:** ``freezer_usage_multiplier`` - **Type:** ``Double`` @@ -5275,7 +5275,7 @@ Whether there is a cooking range/oven present. **Cooking Range/Oven: Location** -The space type for the cooking range/oven location. If not provided, the OS-HPXML default (see HPXML Cooking Range/Oven) is used. +The space type for the cooking range/oven location. If not provided, the OS-HPXML default (see HPXML Cooking Range/Oven) is used. - **Name:** ``cooking_range_oven_location`` - **Type:** ``Choice`` @@ -5301,7 +5301,7 @@ Type of fuel used by the cooking range/oven. **Cooking Range/Oven: Is Induction** -Whether the cooking range is induction. If not provided, the OS-HPXML default (see HPXML Cooking Range/Oven) is used. +Whether the cooking range is induction. If not provided, the OS-HPXML default (see HPXML Cooking Range/Oven) is used. - **Name:** ``cooking_range_oven_is_induction`` - **Type:** ``Boolean`` @@ -5312,7 +5312,7 @@ Whether the cooking range is induction. If not provided, the OS-HPXML default (s **Cooking Range/Oven: Is Convection** -Whether the oven is convection. If not provided, the OS-HPXML default (see HPXML Cooking Range/Oven) is used. +Whether the oven is convection. If not provided, the OS-HPXML default (see HPXML Cooking Range/Oven) is used. - **Name:** ``cooking_range_oven_is_convection`` - **Type:** ``Boolean`` @@ -5323,7 +5323,7 @@ Whether the oven is convection. If not provided, the OS-HPXML default (see HPXML Cooking Range/Oven) is used. +Multiplier on the cooking range/oven energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Cooking Range/Oven) is used. - **Name:** ``cooking_range_oven_usage_multiplier`` - **Type:** ``Double`` @@ -5345,7 +5345,7 @@ Whether there are any ceiling fans. **Ceiling Fan: Label Energy Use** -The label average energy use of the ceiling fan(s). If neither Efficiency nor Label Energy Use provided, the OS-HPXML default (see HPXML Ceiling Fans) is used. +The label average energy use of the ceiling fan(s). If neither Efficiency nor Label Energy Use provided, the OS-HPXML default (see HPXML Ceiling Fans) is used. - **Name:** ``ceiling_fan_label_energy_use`` - **Type:** ``Double`` @@ -5358,7 +5358,7 @@ The label average energy use of the ceiling fan(s). If neither Efficiency nor La **Ceiling Fan: Efficiency** -The efficiency rating of the ceiling fan(s) at medium speed. Only used if Label Energy Use not provided. If neither Efficiency nor Label Energy Use provided, the OS-HPXML default (see HPXML Ceiling Fans) is used. +The efficiency rating of the ceiling fan(s) at medium speed. Only used if Label Energy Use not provided. If neither Efficiency nor Label Energy Use provided, the OS-HPXML default (see HPXML Ceiling Fans) is used. - **Name:** ``ceiling_fan_efficiency`` - **Type:** ``Double`` @@ -5371,7 +5371,7 @@ The efficiency rating of the ceiling fan(s) at medium speed. Only used if Label **Ceiling Fan: Quantity** -Total number of ceiling fans. If not provided, the OS-HPXML default (see HPXML Ceiling Fans) is used. +Total number of ceiling fans. If not provided, the OS-HPXML default (see HPXML Ceiling Fans) is used. - **Name:** ``ceiling_fan_quantity`` - **Type:** ``Integer`` @@ -5384,7 +5384,7 @@ Total number of ceiling fans. If not provided, the OS-HPXML default (see HPXML Ceiling Fans) is used. +The cooling setpoint temperature offset during months when the ceiling fans are operating. Only applies if ceiling fan quantity is greater than zero. If not provided, the OS-HPXML default (see HPXML Ceiling Fans) is used. - **Name:** ``ceiling_fan_cooling_setpoint_temp_offset`` - **Type:** ``Double`` @@ -5408,7 +5408,7 @@ Whether there are televisions. **Misc Plug Loads: Television Annual kWh** -The annual energy consumption of the television plug loads. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +The annual energy consumption of the television plug loads. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_television_annual_kwh`` - **Type:** ``Double`` @@ -5421,7 +5421,7 @@ The annual energy consumption of the television plug loads. If not provided, the **Misc Plug Loads: Television Usage Multiplier** -Multiplier on the television energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +Multiplier on the television energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_television_usage_multiplier`` - **Type:** ``Double`` @@ -5432,7 +5432,7 @@ Multiplier on the television energy usage that can reflect, e.g., high/low usage **Misc Plug Loads: Other Annual kWh** -The annual energy consumption of the other residual plug loads. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +The annual energy consumption of the other residual plug loads. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_other_annual_kwh`` - **Type:** ``Double`` @@ -5445,7 +5445,7 @@ The annual energy consumption of the other residual plug loads. If not provided, **Misc Plug Loads: Other Sensible Fraction** -Fraction of other residual plug loads' internal gains that are sensible. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +Fraction of other residual plug loads' internal gains that are sensible. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_other_frac_sensible`` - **Type:** ``Double`` @@ -5458,7 +5458,7 @@ Fraction of other residual plug loads' internal gains that are sensible. If not **Misc Plug Loads: Other Latent Fraction** -Fraction of other residual plug loads' internal gains that are latent. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +Fraction of other residual plug loads' internal gains that are latent. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_other_frac_latent`` - **Type:** ``Double`` @@ -5471,7 +5471,7 @@ Fraction of other residual plug loads' internal gains that are latent. If not pr **Misc Plug Loads: Other Usage Multiplier** -Multiplier on the other energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +Multiplier on the other energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_other_usage_multiplier`` - **Type:** ``Double`` @@ -5493,7 +5493,7 @@ Whether there is a well pump. **Misc Plug Loads: Well Pump Annual kWh** -The annual energy consumption of the well pump plug loads. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +The annual energy consumption of the well pump plug loads. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_well_pump_annual_kwh`` - **Type:** ``Double`` @@ -5506,7 +5506,7 @@ The annual energy consumption of the well pump plug loads. If not provided, the **Misc Plug Loads: Well Pump Usage Multiplier** -Multiplier on the well pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +Multiplier on the well pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_well_pump_usage_multiplier`` - **Type:** ``Double`` @@ -5528,7 +5528,7 @@ Whether there is an electric vehicle. **Misc Plug Loads: Vehicle Annual kWh** -The annual energy consumption of the electric vehicle plug loads. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +The annual energy consumption of the electric vehicle plug loads. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_vehicle_annual_kwh`` - **Type:** ``Double`` @@ -5541,7 +5541,7 @@ The annual energy consumption of the electric vehicle plug loads. If not provide **Misc Plug Loads: Vehicle Usage Multiplier** -Multiplier on the electric vehicle energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. +Multiplier on the electric vehicle energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Plug Loads) is used. - **Name:** ``misc_plug_loads_vehicle_usage_multiplier`` - **Type:** ``Double`` @@ -5576,7 +5576,7 @@ The fuel type of the fuel loads grill. **Misc Fuel Loads: Grill Annual therm** -The annual energy consumption of the fuel loads grill. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. +The annual energy consumption of the fuel loads grill. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. - **Name:** ``misc_fuel_loads_grill_annual_therm`` - **Type:** ``Double`` @@ -5589,7 +5589,7 @@ The annual energy consumption of the fuel loads grill. If not provided, the OS-H **Misc Fuel Loads: Grill Usage Multiplier** -Multiplier on the fuel loads grill energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. +Multiplier on the fuel loads grill energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. - **Name:** ``misc_fuel_loads_grill_usage_multiplier`` - **Type:** ``Double`` @@ -5624,7 +5624,7 @@ The fuel type of the fuel loads lighting. **Misc Fuel Loads: Lighting Annual therm** -The annual energy consumption of the fuel loads lighting. If not provided, the OS-HPXML default (see HPXML Fuel Loads)is used. +The annual energy consumption of the fuel loads lighting. If not provided, the OS-HPXML default (see HPXML Fuel Loads)is used. - **Name:** ``misc_fuel_loads_lighting_annual_therm`` - **Type:** ``Double`` @@ -5637,7 +5637,7 @@ The annual energy consumption of the fuel loads lighting. If not provided, the O **Misc Fuel Loads: Lighting Usage Multiplier** -Multiplier on the fuel loads lighting energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. +Multiplier on the fuel loads lighting energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. - **Name:** ``misc_fuel_loads_lighting_usage_multiplier`` - **Type:** ``Double`` @@ -5672,7 +5672,7 @@ The fuel type of the fuel loads fireplace. **Misc Fuel Loads: Fireplace Annual therm** -The annual energy consumption of the fuel loads fireplace. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. +The annual energy consumption of the fuel loads fireplace. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. - **Name:** ``misc_fuel_loads_fireplace_annual_therm`` - **Type:** ``Double`` @@ -5685,7 +5685,7 @@ The annual energy consumption of the fuel loads fireplace. If not provided, the **Misc Fuel Loads: Fireplace Sensible Fraction** -Fraction of fireplace residual fuel loads' internal gains that are sensible. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. +Fraction of fireplace residual fuel loads' internal gains that are sensible. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. - **Name:** ``misc_fuel_loads_fireplace_frac_sensible`` - **Type:** ``Double`` @@ -5698,7 +5698,7 @@ Fraction of fireplace residual fuel loads' internal gains that are sensible. If **Misc Fuel Loads: Fireplace Latent Fraction** -Fraction of fireplace residual fuel loads' internal gains that are latent. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. +Fraction of fireplace residual fuel loads' internal gains that are latent. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. - **Name:** ``misc_fuel_loads_fireplace_frac_latent`` - **Type:** ``Double`` @@ -5711,7 +5711,7 @@ Fraction of fireplace residual fuel loads' internal gains that are latent. If no **Misc Fuel Loads: Fireplace Usage Multiplier** -Multiplier on the fuel loads fireplace energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. +Multiplier on the fuel loads fireplace energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see HPXML Fuel Loads) is used. - **Name:** ``misc_fuel_loads_fireplace_usage_multiplier`` - **Type:** ``Double`` @@ -5733,7 +5733,7 @@ Whether there is a pool. **Pool: Pump Annual kWh** -The annual energy consumption of the pool pump. If not provided, the OS-HPXML default (see Pool Pump) is used. +The annual energy consumption of the pool pump. If not provided, the OS-HPXML default (see Pool Pump) is used. - **Name:** ``pool_pump_annual_kwh`` - **Type:** ``Double`` @@ -5746,7 +5746,7 @@ The annual energy consumption of the pool pump. If not provided, the OS-HPXML de **Pool: Pump Usage Multiplier** -Multiplier on the pool pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see Pool Pump) is used. +Multiplier on the pool pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see Pool Pump) is used. - **Name:** ``pool_pump_usage_multiplier`` - **Type:** ``Double`` @@ -5770,7 +5770,7 @@ The type of pool heater. Use 'none' if there is no pool heater. **Pool: Heater Annual kWh** -The annual energy consumption of the electric resistance pool heater. If not provided, the OS-HPXML default (see Pool Heater) is used. +The annual energy consumption of the electric resistance pool heater. If not provided, the OS-HPXML default (see Pool Heater) is used. - **Name:** ``pool_heater_annual_kwh`` - **Type:** ``Double`` @@ -5783,7 +5783,7 @@ The annual energy consumption of the electric resistance pool heater. If not pro **Pool: Heater Annual therm** -The annual energy consumption of the gas fired pool heater. If not provided, the OS-HPXML default (see Pool Heater) is used. +The annual energy consumption of the gas fired pool heater. If not provided, the OS-HPXML default (see Pool Heater) is used. - **Name:** ``pool_heater_annual_therm`` - **Type:** ``Double`` @@ -5796,7 +5796,7 @@ The annual energy consumption of the gas fired pool heater. If not provided, the **Pool: Heater Usage Multiplier** -Multiplier on the pool heater energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see Pool Heater) is used. +Multiplier on the pool heater energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see Pool Heater) is used. - **Name:** ``pool_heater_usage_multiplier`` - **Type:** ``Double`` @@ -5818,7 +5818,7 @@ Whether there is a permanent spa. **Permanent Spa: Pump Annual kWh** -The annual energy consumption of the permanent spa pump. If not provided, the OS-HPXML default (see Permanent Spa Pump) is used. +The annual energy consumption of the permanent spa pump. If not provided, the OS-HPXML default (see Permanent Spa Pump) is used. - **Name:** ``permanent_spa_pump_annual_kwh`` - **Type:** ``Double`` @@ -5831,7 +5831,7 @@ The annual energy consumption of the permanent spa pump. If not provided, the OS **Permanent Spa: Pump Usage Multiplier** -Multiplier on the permanent spa pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see Permanent Spa Pump) is used. +Multiplier on the permanent spa pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see Permanent Spa Pump) is used. - **Name:** ``permanent_spa_pump_usage_multiplier`` - **Type:** ``Double`` @@ -5855,7 +5855,7 @@ The type of permanent spa heater. Use 'none' if there is no permanent spa heater **Permanent Spa: Heater Annual kWh** -The annual energy consumption of the electric resistance permanent spa heater. If not provided, the OS-HPXML default (see Permanent Spa Heater) is used. +The annual energy consumption of the electric resistance permanent spa heater. If not provided, the OS-HPXML default (see Permanent Spa Heater) is used. - **Name:** ``permanent_spa_heater_annual_kwh`` - **Type:** ``Double`` @@ -5868,7 +5868,7 @@ The annual energy consumption of the electric resistance permanent spa heater. I **Permanent Spa: Heater Annual therm** -The annual energy consumption of the gas fired permanent spa heater. If not provided, the OS-HPXML default (see Permanent Spa Heater) is used. +The annual energy consumption of the gas fired permanent spa heater. If not provided, the OS-HPXML default (see Permanent Spa Heater) is used. - **Name:** ``permanent_spa_heater_annual_therm`` - **Type:** ``Double`` @@ -5881,7 +5881,7 @@ The annual energy consumption of the gas fired permanent spa heater. If not prov **Permanent Spa: Heater Usage Multiplier** -Multiplier on the permanent spa heater energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see Permanent Spa Heater) is used. +Multiplier on the permanent spa heater energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see Permanent Spa Heater) is used. - **Name:** ``permanent_spa_heater_usage_multiplier`` - **Type:** ``Double`` diff --git a/BuildResidentialHPXML/measure.rb b/BuildResidentialHPXML/measure.rb index bd3faca4f7..b45579f0c6 100644 --- a/BuildResidentialHPXML/measure.rb +++ b/BuildResidentialHPXML/measure.rb @@ -343,7 +343,7 @@ def arguments(model) # rubocop:disable Lint/UnusedMethodArgument arg = OpenStudio::Measure::OSArgument::makeDoubleArgument('geometry_unit_num_occupants', false) arg.setDisplayName('Geometry: Unit Number of Occupants') arg.setUnits('#') - arg.setDescription('The number of occupants in the unit. If not provided, an *asset* calculation is performed assuming standard occupancy, in which various end use defaults (e.g., plug loads, appliances, and hot water usage) are calculated based on Number of Bedrooms and Conditioned Floor Area per ANSI/RESNET/ICC 301-2019. If provided, an *operational* calculation is instead performed in which the end use defaults are adjusted using the relationship between Number of Bedrooms and Number of Occupants from RECS 2015.') + arg.setDescription('The number of occupants in the unit. If not provided, an *asset* calculation is performed assuming standard occupancy, in which various end use defaults (e.g., plug loads, appliances, and hot water usage) are calculated based on Number of Bedrooms and Conditioned Floor Area per ANSI/RESNET/ICC 301. If provided, an *operational* calculation is instead performed in which the end use defaults to reflect real-world data (where possible).') args << arg arg = OpenStudio::Measure::OSArgument::makeIntegerArgument('geometry_building_num_units', false) @@ -4082,13 +4082,14 @@ def self.create_geometry_envelope(runner, model, args) return false end - if args[:geometry_unit_type] == HPXML::ResidentialTypeSFD + case args[:geometry_unit_type] + when HPXML::ResidentialTypeSFD success = Geometry.create_single_family_detached(runner: runner, model: model, **args) - elsif args[:geometry_unit_type] == HPXML::ResidentialTypeSFA + when HPXML::ResidentialTypeSFA success = Geometry.create_single_family_attached(model: model, **args) - elsif args[:geometry_unit_type] == HPXML::ResidentialTypeApartment + when HPXML::ResidentialTypeApartment success = Geometry.create_apartment(model: model, **args) - elsif args[:geometry_unit_type] == HPXML::ResidentialTypeManufactured + when HPXML::ResidentialTypeManufactured success = Geometry.create_single_family_detached(runner: runner, model: model, **args) end return false if not success @@ -4613,7 +4614,8 @@ def self.set_site(hpxml_bldg, args) adb_walls = [args[:geometry_unit_left_wall_is_adiabatic], args[:geometry_unit_right_wall_is_adiabatic], args[:geometry_unit_front_wall_is_adiabatic], args[:geometry_unit_back_wall_is_adiabatic]] n_walls_attached = adb_walls.count(true) - if [HPXML::ResidentialTypeSFA, HPXML::ResidentialTypeApartment].include? args[:geometry_unit_type] + case args[:geometry_unit_type] + when HPXML::ResidentialTypeSFA, HPXML::ResidentialTypeApartment if n_walls_attached == 3 hpxml_bldg.site.surroundings = HPXML::SurroundingsThreeSides elsif n_walls_attached == 2 @@ -4636,7 +4638,7 @@ def self.set_site(hpxml_bldg, args) hpxml_bldg.site.vertical_surroundings = HPXML::VerticalSurroundingsNoAboveOrBelow end end - elsif [HPXML::ResidentialTypeSFD, HPXML::ResidentialTypeManufactured].include? args[:geometry_unit_type] + when HPXML::ResidentialTypeSFD, HPXML::ResidentialTypeManufactured hpxml_bldg.site.surroundings = HPXML::SurroundingsStandAlone hpxml_bldg.site.vertical_surroundings = HPXML::VerticalSurroundingsNoAboveOrBelow end @@ -5178,9 +5180,10 @@ def self.set_floors(hpxml_bldg, args, sorted_surfaces) @surface_ids[surface.name.to_s] = hpxml_bldg.floors[-1].id if hpxml_bldg.floors[-1].is_thermal_boundary - if [HPXML::LocationAtticUnvented, HPXML::LocationAtticVented].include? exterior_adjacent_to + case exterior_adjacent_to + when HPXML::LocationAtticUnvented, HPXML::LocationAtticVented hpxml_bldg.floors[-1].insulation_assembly_r_value = args[:ceiling_assembly_r] - elsif [HPXML::LocationGarage].include? exterior_adjacent_to + when HPXML::LocationGarage hpxml_bldg.floors[-1].insulation_assembly_r_value = args[:floor_over_garage_assembly_r] else hpxml_bldg.floors[-1].insulation_assembly_r_value = args[:floor_over_foundation_assembly_r] @@ -5229,11 +5232,7 @@ def self.set_slabs(hpxml_bldg, model, args, sorted_surfaces) exposed_perimeter = Geometry.calculate_exposed_perimeter(model: model, ground_floor_surfaces: [surface], has_foundation_walls: has_foundation_walls).round(1) next if exposed_perimeter == 0 - if [HPXML::LocationCrawlspaceVented, - HPXML::LocationCrawlspaceUnvented, - HPXML::LocationCrawlspaceConditioned, - HPXML::LocationBasementUnconditioned, - HPXML::LocationBasementConditioned].include? interior_adjacent_to + if has_foundation_walls exposed_perimeter -= Geometry.get_unexposed_garage_perimeter(**args) end @@ -5627,13 +5626,14 @@ def self.set_cooling_systems(hpxml_bldg, args) end if cooling_system_type != HPXML::HVACTypeEvaporativeCooler - if args[:cooling_system_cooling_efficiency_type] == HPXML::UnitsSEER + case args[:cooling_system_cooling_efficiency_type] + when HPXML::UnitsSEER cooling_efficiency_seer = args[:cooling_system_cooling_efficiency] - elsif args[:cooling_system_cooling_efficiency_type] == HPXML::UnitsSEER2 + when HPXML::UnitsSEER2 cooling_efficiency_seer2 = args[:cooling_system_cooling_efficiency] - elsif args[:cooling_system_cooling_efficiency_type] == HPXML::UnitsEER + when HPXML::UnitsEER cooling_efficiency_eer = args[:cooling_system_cooling_efficiency] - elsif args[:cooling_system_cooling_efficiency_type] == HPXML::UnitsCEER + when HPXML::UnitsCEER cooling_efficiency_ceer = args[:cooling_system_cooling_efficiency] end end @@ -5695,10 +5695,11 @@ def self.set_cooling_systems(hpxml_bldg, args) cooling_perf_data_data_points.each do |cooling_perf_data_data_point| outdoor_temperature, min_speed_cap_or_frac, max_speed_cap_or_frac, min_speed_cop, max_speed_cop = cooling_perf_data_data_point - if hvac_perf_data_capacity_type == 'Absolute capacities' + case hvac_perf_data_capacity_type + when 'Absolute capacities' min_speed_capacity = Float(min_speed_cap_or_frac) max_speed_capacity = Float(max_speed_cap_or_frac) - elsif hvac_perf_data_capacity_type == 'Normalized capacity fractions' + when 'Normalized capacity fractions' min_speed_capacity_fraction_of_nominal = Float(min_speed_cap_or_frac) max_speed_capacity_fraction_of_nominal = Float(max_speed_cap_or_frac) end @@ -5737,7 +5738,8 @@ def self.set_heat_pumps(hpxml_bldg, args) return if heat_pump_type == Constants::None - if args[:heat_pump_backup_type] == HPXML::HeatPumpBackupTypeIntegrated + case args[:heat_pump_backup_type] + when HPXML::HeatPumpBackupTypeIntegrated backup_type = args[:heat_pump_backup_type] backup_heating_fuel = args[:heat_pump_backup_fuel] backup_heating_capacity = args[:heat_pump_backup_heating_capacity] @@ -5747,7 +5749,7 @@ def self.set_heat_pumps(hpxml_bldg, args) else backup_heating_efficiency_afue = args[:heat_pump_backup_heating_efficiency] end - elsif args[:heat_pump_backup_type] == HPXML::HeatPumpBackupTypeSeparate + when HPXML::HeatPumpBackupTypeSeparate if args[:heating_system_2_type] == Constants::None fail "Heat pump backup type specified as '#{args[:heat_pump_backup_type]}' but no heating system provided." end @@ -5769,19 +5771,21 @@ def self.set_heat_pumps(hpxml_bldg, args) compressor_type = args[:heat_pump_cooling_compressor_type] end - if args[:heat_pump_heating_efficiency_type] == HPXML::UnitsHSPF + case args[:heat_pump_heating_efficiency_type] + when HPXML::UnitsHSPF heating_efficiency_hspf = args[:heat_pump_heating_efficiency] - elsif args[:heat_pump_heating_efficiency_type] == HPXML::UnitsHSPF2 + when HPXML::UnitsHSPF2 heating_efficiency_hspf2 = args[:heat_pump_heating_efficiency] - elsif args[:heat_pump_heating_efficiency_type] == HPXML::UnitsCOP + when HPXML::UnitsCOP heating_efficiency_cop = args[:heat_pump_heating_efficiency] end - if args[:heat_pump_cooling_efficiency_type] == HPXML::UnitsSEER + case args[:heat_pump_cooling_efficiency_type] + when HPXML::UnitsSEER cooling_efficiency_seer = args[:heat_pump_cooling_efficiency] - elsif args[:heat_pump_cooling_efficiency_type] == HPXML::UnitsSEER2 + when HPXML::UnitsSEER2 cooling_efficiency_seer2 = args[:heat_pump_cooling_efficiency] - elsif args[:heat_pump_cooling_efficiency_type] == HPXML::UnitsEER + when HPXML::UnitsEER cooling_efficiency_eer = args[:heat_pump_cooling_efficiency] end @@ -5848,10 +5852,11 @@ def self.set_heat_pumps(hpxml_bldg, args) heating_perf_data_data_points.each do |heating_perf_data_data_point| outdoor_temperature, min_speed_cap_or_frac, max_speed_cap_or_frac, min_speed_cop, max_speed_cop = heating_perf_data_data_point - if hvac_perf_data_capacity_type == 'Absolute capacities' + case hvac_perf_data_capacity_type + when 'Absolute capacities' min_speed_capacity = Float(min_speed_cap_or_frac) max_speed_capacity = Float(max_speed_cap_or_frac) - elsif hvac_perf_data_capacity_type == 'Normalized capacity fractions' + when 'Normalized capacity fractions' min_speed_capacity_fraction_of_nominal = Float(min_speed_cap_or_frac) max_speed_capacity_fraction_of_nominal = Float(max_speed_cap_or_frac) end @@ -5885,10 +5890,11 @@ def self.set_heat_pumps(hpxml_bldg, args) cooling_perf_data_data_points.each do |cooling_perf_data_data_point| outdoor_temperature, min_speed_cap_or_frac, max_speed_cap_or_frac, min_speed_cop, max_speed_cop = cooling_perf_data_data_point - if hvac_perf_data_capacity_type == 'Absolute capacities' + case hvac_perf_data_capacity_type + when 'Absolute capacities' min_speed_capacity = Float(min_speed_cap_or_frac) max_speed_capacity = Float(max_speed_cap_or_frac) - elsif hvac_perf_data_capacity_type == 'Normalized capacity fractions' + when 'Normalized capacity fractions' min_speed_capacity_fraction_of_nominal = Float(min_speed_cap_or_frac) max_speed_capacity_fraction_of_nominal = Float(max_speed_cap_or_frac) end @@ -5926,12 +5932,12 @@ def self.set_geothermal_loop(hpxml_bldg, args) return if args[:geothermal_loop_configuration].nil? || args[:geothermal_loop_configuration] == Constants::None if not args[:geothermal_loop_pipe_diameter].nil? - pipe_diameter = args[:geothermal_loop_pipe_diameter] - if pipe_diameter == '3/4" pipe' + case args[:geothermal_loop_pipe_diameter] + when '3/4" pipe' pipe_diameter = 0.75 - elsif pipe_diameter == '1" pipe' + when '1" pipe' pipe_diameter = 1.0 - elsif pipe_diameter == '1-1/4" pipe' + when '1-1/4" pipe' pipe_diameter = 1.25 end end @@ -6018,24 +6024,25 @@ def self.set_hvac_distribution(hpxml_bldg, args) # AirDistribution? air_distribution_systems = [] hpxml_bldg.heating_systems.each do |heating_system| - if [HPXML::HVACTypeFurnace].include?(heating_system.heating_system_type) + case heating_system.heating_system_type + when HPXML::HVACTypeFurnace air_distribution_systems << heating_system end end hpxml_bldg.cooling_systems.each do |cooling_system| - if [HPXML::HVACTypeCentralAirConditioner].include?(cooling_system.cooling_system_type) - air_distribution_systems << cooling_system - elsif [HPXML::HVACTypeEvaporativeCooler, HPXML::HVACTypeMiniSplitAirConditioner].include?(cooling_system.cooling_system_type) && args[:cooling_system_is_ducted] + case cooling_system.cooling_system_type + when HPXML::HVACTypeCentralAirConditioner air_distribution_systems << cooling_system + when HPXML::HVACTypeEvaporativeCooler, HPXML::HVACTypeMiniSplitAirConditioner + air_distribution_systems << cooling_system if args[:cooling_system_is_ducted] end end hpxml_bldg.heat_pumps.each do |heat_pump| - if [HPXML::HVACTypeHeatPumpAirToAir, HPXML::HVACTypeHeatPumpGroundToAir].include? heat_pump.heat_pump_type + case heat_pump.heat_pump_type + when HPXML::HVACTypeHeatPumpAirToAir, HPXML::HVACTypeHeatPumpGroundToAir air_distribution_systems << heat_pump - elsif [HPXML::HVACTypeHeatPumpMiniSplit].include?(heat_pump.heat_pump_type) - if args[:heat_pump_is_ducted] - air_distribution_systems << heat_pump if args[:heat_pump_is_ducted] - end + when HPXML::HVACTypeHeatPumpMiniSplit + air_distribution_systems << heat_pump if args[:heat_pump_is_ducted] end end @@ -6089,7 +6096,8 @@ def self.set_hvac_distribution(hpxml_bldg, args) def self.set_hvac_blower(hpxml_bldg, args) # Blower fan W/cfm hpxml_bldg.hvac_systems.each do |hvac_system| - next unless (!hvac_system.distribution_system.nil? && hvac_system.distribution_system.distribution_system_type == HPXML::HVACDistributionTypeAir) || (hvac_system.is_a?(HPXML::HeatPump) && [HPXML::HVACTypeHeatPumpMiniSplit].include?(hvac_system.heat_pump_type)) + next unless (!hvac_system.distribution_system.nil? && hvac_system.distribution_system.distribution_system_type == HPXML::HVACDistributionTypeAir) || + (hvac_system.is_a?(HPXML::HeatPump) && hvac_system.heat_pump_type == HPXML::HVACTypeHeatPumpMiniSplit) fan_watts_per_cfm = args[:hvac_blower_fan_watts_per_cfm] @@ -6098,11 +6106,14 @@ def self.set_hvac_blower(hpxml_bldg, args) hvac_system.fan_watts_per_cfm = fan_watts_per_cfm end elsif hvac_system.is_a?(HPXML::CoolingSystem) - if [HPXML::HVACTypeCentralAirConditioner, HPXML::HVACTypeMiniSplitAirConditioner].include?(hvac_system.cooling_system_type) + if [HPXML::HVACTypeCentralAirConditioner, + HPXML::HVACTypeMiniSplitAirConditioner].include?(hvac_system.cooling_system_type) hvac_system.fan_watts_per_cfm = fan_watts_per_cfm end elsif hvac_system.is_a?(HPXML::HeatPump) - if [HPXML::HVACTypeHeatPumpAirToAir, HPXML::HVACTypeHeatPumpMiniSplit, HPXML::HVACTypeHeatPumpGroundToAir].include?(hvac_system.heat_pump_type) + if [HPXML::HVACTypeHeatPumpAirToAir, + HPXML::HVACTypeHeatPumpMiniSplit, + HPXML::HVACTypeHeatPumpGroundToAir].include?(hvac_system.heat_pump_type) hvac_system.fan_watts_per_cfm = fan_watts_per_cfm end end @@ -6138,21 +6149,23 @@ def self.get_location(location, foundation_type, attic_type) return if location.nil? if location == HPXML::LocationCrawlspace - if foundation_type == HPXML::FoundationTypeCrawlspaceUnvented + case foundation_type + when HPXML::FoundationTypeCrawlspaceUnvented return HPXML::LocationCrawlspaceUnvented - elsif foundation_type == HPXML::FoundationTypeCrawlspaceVented + when HPXML::FoundationTypeCrawlspaceVented return HPXML::LocationCrawlspaceVented - elsif foundation_type == HPXML::FoundationTypeCrawlspaceConditioned + when HPXML::FoundationTypeCrawlspaceConditioned return HPXML::LocationCrawlspaceConditioned else fail "Specified '#{location}' but foundation type is '#{foundation_type}'." end elsif location == HPXML::LocationAttic - if attic_type == HPXML::AtticTypeUnvented + case attic_type + when HPXML::AtticTypeUnvented return HPXML::LocationAtticUnvented - elsif attic_type == HPXML::AtticTypeVented + when HPXML::AtticTypeVented return HPXML::LocationAtticVented - elsif attic_type == HPXML::AtticTypeConditioned + when HPXML::AtticTypeConditioned return HPXML::LocationConditionedSpace else fail "Specified '#{location}' but attic type is '#{attic_type}'." @@ -6393,24 +6406,26 @@ def self.set_hvac_control(hpxml, hpxml_bldg, args, weather) def self.set_ventilation_fans(hpxml_bldg, args) if args[:mech_vent_fan_type] != Constants::None - if [HPXML::MechVentTypeERV].include?(args[:mech_vent_fan_type]) - if args[:mech_vent_recovery_efficiency_type] == 'Unadjusted' + distribution_system_idref = nil + + case args[:mech_vent_fan_type] + when HPXML::MechVentTypeERV + case args[:mech_vent_recovery_efficiency_type] + when 'Unadjusted' total_recovery_efficiency = args[:mech_vent_total_recovery_efficiency] sensible_recovery_efficiency = args[:mech_vent_sensible_recovery_efficiency] - elsif args[:mech_vent_recovery_efficiency_type] == 'Adjusted' + when 'Adjusted' total_recovery_efficiency_adjusted = args[:mech_vent_total_recovery_efficiency] sensible_recovery_efficiency_adjusted = args[:mech_vent_sensible_recovery_efficiency] end - elsif [HPXML::MechVentTypeHRV].include?(args[:mech_vent_fan_type]) - if args[:mech_vent_recovery_efficiency_type] == 'Unadjusted' + when HPXML::MechVentTypeHRV + case args[:mech_vent_recovery_efficiency_type] + when 'Unadjusted' sensible_recovery_efficiency = args[:mech_vent_sensible_recovery_efficiency] - elsif args[:mech_vent_recovery_efficiency_type] == 'Adjusted' + when 'Adjusted' sensible_recovery_efficiency_adjusted = args[:mech_vent_sensible_recovery_efficiency] end - end - - distribution_system_idref = nil - if args[:mech_vent_fan_type] == HPXML::MechVentTypeCFIS + when HPXML::MechVentTypeCFIS hpxml_bldg.hvac_distributions.each do |hvac_distribution| next unless hvac_distribution.distribution_system_type == HPXML::HVACDistributionTypeAir next if hvac_distribution.air_type != HPXML::AirTypeRegularVelocity @@ -6474,19 +6489,21 @@ def self.set_ventilation_fans(hpxml_bldg, args) if args[:mech_vent_2_fan_type] != Constants::None - if [HPXML::MechVentTypeERV].include?(args[:mech_vent_2_fan_type]) - - if args[:mech_vent_2_recovery_efficiency_type] == 'Unadjusted' + case args[:mech_vent_2_fan_type] + when HPXML::MechVentTypeERV + case args[:mech_vent_2_recovery_efficiency_type] + when 'Unadjusted' total_recovery_efficiency = args[:mech_vent_2_total_recovery_efficiency] sensible_recovery_efficiency = args[:mech_vent_2_sensible_recovery_efficiency] - elsif args[:mech_vent_2_recovery_efficiency_type] == 'Adjusted' + when 'Adjusted' total_recovery_efficiency_adjusted = args[:mech_vent_2_total_recovery_efficiency] sensible_recovery_efficiency_adjusted = args[:mech_vent_2_sensible_recovery_efficiency] end - elsif [HPXML::MechVentTypeHRV].include?(args[:mech_vent_2_fan_type]) - if args[:mech_vent_2_recovery_efficiency_type] == 'Unadjusted' + when HPXML::MechVentTypeHRV + case args[:mech_vent_2_recovery_efficiency_type] + when 'Unadjusted' sensible_recovery_efficiency = args[:mech_vent_2_sensible_recovery_efficiency] - elsif args[:mech_vent_2_recovery_efficiency_type] == 'Adjusted' + when 'Adjusted' sensible_recovery_efficiency_adjusted = args[:mech_vent_2_sensible_recovery_efficiency] end end @@ -6560,9 +6577,10 @@ def self.set_water_heating_systems(hpxml_bldg, args) location = get_location(args[:water_heater_location], hpxml_bldg.foundations[-1].foundation_type, hpxml_bldg.attics[-1].attic_type) if not [HPXML::WaterHeaterTypeCombiStorage, HPXML::WaterHeaterTypeCombiTankless].include? water_heater_type - if args[:water_heater_efficiency_type] == 'EnergyFactor' + case args[:water_heater_efficiency_type] + when 'EnergyFactor' energy_factor = args[:water_heater_efficiency] - elsif args[:water_heater_efficiency_type] == 'UniformEnergyFactor' + when 'UniformEnergyFactor' uniform_energy_factor = args[:water_heater_efficiency] if water_heater_type != HPXML::WaterHeaterTypeTankless usage_bin = args[:water_heater_usage_bin] @@ -6955,9 +6973,10 @@ def self.set_lighting(hpxml_bldg, args) def self.set_dehumidifier(hpxml_bldg, args) return if args[:dehumidifier_type] == Constants::None - if args[:dehumidifier_efficiency_type] == 'EnergyFactor' + case args[:dehumidifier_efficiency_type] + when 'EnergyFactor' energy_factor = args[:dehumidifier_efficiency] - elsif args[:dehumidifier_efficiency_type] == 'IntegratedEnergyFactor' + when 'IntegratedEnergyFactor' integrated_energy_factor = args[:dehumidifier_efficiency] end @@ -6987,9 +7006,10 @@ def self.set_clothes_washer(hpxml_bldg, args) return if args[:water_heater_type] == Constants::None return unless args[:clothes_washer_present] - if args[:clothes_washer_efficiency_type] == 'ModifiedEnergyFactor' + case args[:clothes_washer_efficiency_type] + when 'ModifiedEnergyFactor' modified_energy_factor = args[:clothes_washer_efficiency] - elsif args[:clothes_washer_efficiency_type] == 'IntegratedModifiedEnergyFactor' + when 'IntegratedModifiedEnergyFactor' integrated_modified_energy_factor = args[:clothes_washer_efficiency] end @@ -7021,9 +7041,10 @@ def self.set_clothes_dryer(hpxml_bldg, args) return unless args[:clothes_washer_present] return unless args[:clothes_dryer_present] - if args[:clothes_dryer_efficiency_type] == 'EnergyFactor' + case args[:clothes_dryer_efficiency_type] + when 'EnergyFactor' energy_factor = args[:clothes_dryer_efficiency] - elsif args[:clothes_dryer_efficiency_type] == 'CombinedEnergyFactor' + when 'CombinedEnergyFactor' combined_energy_factor = args[:clothes_dryer_efficiency] end @@ -7061,9 +7082,10 @@ def self.set_dishwasher(hpxml_bldg, args) return if args[:water_heater_type] == Constants::None return unless args[:dishwasher_present] - if args[:dishwasher_efficiency_type] == 'RatedAnnualkWh' + case args[:dishwasher_efficiency_type] + when 'RatedAnnualkWh' rated_annual_kwh = args[:dishwasher_efficiency] - elsif args[:dishwasher_efficiency_type] == 'EnergyFactor' + when 'EnergyFactor' energy_factor = args[:dishwasher_efficiency] end @@ -7305,14 +7327,13 @@ def self.set_misc_fuel_loads_fireplace(hpxml_bldg, args) def self.set_pool(hpxml_bldg, args) return unless args[:pool_present] - if [HPXML::HeaterTypeElectricResistance, HPXML::HeaterTypeHeatPump].include?(args[:pool_heater_type]) + case args[:pool_heater_type] + when HPXML::HeaterTypeElectricResistance, HPXML::HeaterTypeHeatPump if not args[:pool_heater_annual_kwh].nil? heater_load_units = HPXML::UnitsKwhPerYear heater_load_value = args[:pool_heater_annual_kwh] end - end - - if [HPXML::HeaterTypeGas].include?(args[:pool_heater_type]) + when HPXML::HeaterTypeGas if not args[:pool_heater_annual_therm].nil? heater_load_units = HPXML::UnitsThermPerYear heater_load_value = args[:pool_heater_annual_therm] @@ -7343,14 +7364,13 @@ def self.set_pool(hpxml_bldg, args) def self.set_permanent_spa(hpxml_bldg, args) return unless args[:permanent_spa_present] - if [HPXML::HeaterTypeElectricResistance, HPXML::HeaterTypeHeatPump].include?(args[:permanent_spa_heater_type]) + case args[:permanent_spa_heater_type] + when HPXML::HeaterTypeElectricResistance, HPXML::HeaterTypeHeatPump if not args[:permanent_spa_heater_annual_kwh].nil? heater_load_units = HPXML::UnitsKwhPerYear heater_load_value = args[:permanent_spa_heater_annual_kwh] end - end - - if [HPXML::HeaterTypeGas].include?(args[:permanent_spa_heater_type]) + when HPXML::HeaterTypeGas if not args[:permanent_spa_heater_annual_therm].nil? heater_load_units = HPXML::UnitsThermPerYear heater_load_value = args[:permanent_spa_heater_annual_therm] diff --git a/BuildResidentialHPXML/measure.xml b/BuildResidentialHPXML/measure.xml index ea8c333d99..154e205c1e 100644 --- a/BuildResidentialHPXML/measure.xml +++ b/BuildResidentialHPXML/measure.xml @@ -3,8 +3,8 @@ 3.1 build_residential_hpxml a13a8983-2b01-4930-8af2-42030b6e4233 - 6024806a-1640-4c75-86e4-d00329c89e3a - 2024-10-31T22:43:54Z + 32bd49ca-09dc-40ec-b676-f01d76e98566 + 2024-12-09T21:40:28Z 2C38F48B BuildResidentialHPXML HPXML Builder @@ -88,7 +88,7 @@ schedules_unavailable_period_window_natvent_availabilities Schedules: Unavailable Period Window Natural Ventilation Availabilities - The availability of the natural ventilation schedule during unavailable periods. Valid choices are: regular schedule, always available, always unavailable. If multiple periods, use a comma-separated list. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-unavailable-periods'>HPXML Unavailable Periods</a>) is used. + The availability of the natural ventilation schedule during unavailable periods. Valid choices are: regular schedule, always available, always unavailable. If multiple periods, use a comma-separated list. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-unavailable-periods'>HPXML Unavailable Periods</a>) is used. String false false @@ -96,7 +96,7 @@ simulation_control_timestep Simulation Control: Timestep - Value must be a divisor of 60. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. + Value must be a divisor of 60. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. Integer min false @@ -105,7 +105,7 @@ simulation_control_run_period Simulation Control: Run Period - Enter a date range like 'Jan 1 - Dec 31'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. + Enter a date range like 'Jan 1 - Dec 31'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. String false false @@ -113,7 +113,7 @@ simulation_control_run_period_calendar_year Simulation Control: Run Period Calendar Year - This numeric field should contain the calendar year that determines the start day of week. If you are running simulations using AMY weather files, the value entered for calendar year will not be used; it will be overridden by the actual year found in the AMY weather file. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. + This numeric field should contain the calendar year that determines the start day of week. If you are running simulations using AMY weather files, the value entered for calendar year will not be used; it will be overridden by the actual year found in the AMY weather file. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. Integer year false @@ -122,7 +122,7 @@ simulation_control_daylight_saving_enabled Simulation Control: Daylight Saving Enabled - Whether to use daylight saving. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-building-site'>HPXML Building Site</a>) is used. + Whether to use daylight saving. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-building-site'>HPXML Building Site</a>) is used. Boolean false false @@ -140,7 +140,7 @@ simulation_control_daylight_saving_period Simulation Control: Daylight Saving Period - Enter a date range like 'Mar 15 - Dec 15'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-building-site'>HPXML Building Site</a>) is used. + Enter a date range like 'Mar 15 - Dec 15'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-building-site'>HPXML Building Site</a>) is used. String false false @@ -148,7 +148,7 @@ simulation_control_temperature_capacitance_multiplier Simulation Control: Temperature Capacitance Multiplier - Affects the transient calculation of indoor air temperatures. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. + Affects the transient calculation of indoor air temperatures. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. String false false @@ -156,7 +156,7 @@ simulation_control_defrost_model_type Simulation Control: Defrost Model Type - Research feature to select the type of defrost model. Use standard for default E+ defrost setting. Use advanced for an improved model that better accounts for load and energy use during defrost; using advanced may impact simulation runtime. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. + Research feature to select the type of defrost model. Use standard for default E+ defrost setting. Use advanced for an improved model that better accounts for load and energy use during defrost; using advanced may impact simulation runtime. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-simulation-control'>HPXML Simulation Control</a>) is used. Choice false false @@ -192,7 +192,7 @@ site_type Site: Type - The type of site. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. + The type of site. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. Choice false false @@ -214,7 +214,7 @@ site_shielding_of_home Site: Shielding of Home - Presence of nearby buildings, trees, obstructions for infiltration model. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. + Presence of nearby buildings, trees, obstructions for infiltration model. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. Choice false false @@ -236,7 +236,7 @@ site_soil_and_moisture_type Site: Soil and Moisture Type - Type of soil and moisture. This is used to inform ground conductivity and diffusivity. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. + Type of soil and moisture. This is used to inform ground conductivity and diffusivity. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. Choice false false @@ -434,7 +434,7 @@ site_state_code Site: State Code - State code of the home address. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. + State code of the home address. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. Choice false false @@ -656,7 +656,7 @@ site_time_zone_utc_offset Site: Time Zone UTC Offset - Time zone UTC offset of the home address. Must be between -12 and 14. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. + Time zone UTC offset of the home address. Must be between -12 and 14. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. Double hr false @@ -665,7 +665,7 @@ site_elevation Site: Elevation - Elevation of the home address. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. + Elevation of the home address. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. Double ft false @@ -674,7 +674,7 @@ site_latitude Site: Latitude - Latitude of the home address. Must be between -90 and 90. Use negative values for southern hemisphere. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. + Latitude of the home address. Must be between -90 and 90. Use negative values for southern hemisphere. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. Double deg false @@ -683,7 +683,7 @@ site_longitude Site: Longitude - Longitude of the home address. Must be between -180 and 180. Use negative values for the western hemisphere. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. + Longitude of the home address. Must be between -180 and 180. Use negative values for the western hemisphere. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-site'>HPXML Site</a>) is used. Double deg false @@ -869,7 +869,7 @@ geometry_unit_num_bathrooms Geometry: Unit Number of Bathrooms - The number of bathrooms in the unit. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-building-construction'>HPXML Building Construction</a>) is used. + The number of bathrooms in the unit. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-building-construction'>HPXML Building Construction</a>) is used. Integer # false @@ -878,7 +878,7 @@ geometry_unit_num_occupants Geometry: Unit Number of Occupants - The number of occupants in the unit. If not provided, an *asset* calculation is performed assuming standard occupancy, in which various end use defaults (e.g., plug loads, appliances, and hot water usage) are calculated based on Number of Bedrooms and Conditioned Floor Area per ANSI/RESNET/ICC 301-2019. If provided, an *operational* calculation is instead performed in which the end use defaults are adjusted using the relationship between Number of Bedrooms and Number of Occupants from RECS 2015. + The number of occupants in the unit. If not provided, an *asset* calculation is performed assuming standard occupancy, in which various end use defaults (e.g., plug loads, appliances, and hot water usage) are calculated based on Number of Bedrooms and Conditioned Floor Area per ANSI/RESNET/ICC 301. If provided, an *operational* calculation is instead performed in which the end use defaults to reflect real-world data (where possible). Double # false @@ -906,7 +906,7 @@ geometry_unit_height_above_grade Geometry: Unit Height Above Grade - Describes the above-grade height of apartment units on upper floors or homes above ambient or belly-and-wing foundations. It is defined as the height of the lowest conditioned floor above grade and is used to calculate the wind speed for the infiltration model. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-building-construction'>HPXML Building Construction</a>) is used. + Describes the above-grade height of apartment units on upper floors or homes above ambient or belly-and-wing foundations. It is defined as the height of the lowest conditioned floor above grade and is used to calculate the wind speed for the infiltration model. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-building-construction'>HPXML Building Construction</a>) is used. Double ft false @@ -1203,7 +1203,7 @@ neighbor_front_height Neighbor: Front Height - The height of the neighboring building to the front. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-neighbor-buildings'>HPXML Neighbor Building</a>) is used. + The height of the neighboring building to the front. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-neighbor-buildings'>HPXML Neighbor Building</a>) is used. Double ft false @@ -1212,7 +1212,7 @@ neighbor_back_height Neighbor: Back Height - The height of the neighboring building to the back. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-neighbor-buildings'>HPXML Neighbor Building</a>) is used. + The height of the neighboring building to the back. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-neighbor-buildings'>HPXML Neighbor Building</a>) is used. Double ft false @@ -1221,7 +1221,7 @@ neighbor_left_height Neighbor: Left Height - The height of the neighboring building to the left. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-neighbor-buildings'>HPXML Neighbor Building</a>) is used. + The height of the neighboring building to the left. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-neighbor-buildings'>HPXML Neighbor Building</a>) is used. Double ft false @@ -1230,7 +1230,7 @@ neighbor_right_height Neighbor: Right Height - The height of the neighboring building to the right. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-neighbor-buildings'>HPXML Neighbor Building</a>) is used. + The height of the neighboring building to the right. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-neighbor-buildings'>HPXML Neighbor Building</a>) is used. Double ft false @@ -1286,7 +1286,7 @@ foundation_wall_type Foundation Wall: Type - The material type of the foundation wall. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-foundation-walls'>HPXML Foundation Walls</a>) is used. + The material type of the foundation wall. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-foundation-walls'>HPXML Foundation Walls</a>) is used. Choice false false @@ -1328,7 +1328,7 @@ foundation_wall_thickness Foundation Wall: Thickness - The thickness of the foundation wall. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-foundation-walls'>HPXML Foundation Walls</a>) is used. + The thickness of the foundation wall. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-foundation-walls'>HPXML Foundation Walls</a>) is used. Double in false @@ -1367,7 +1367,7 @@ foundation_wall_insulation_distance_to_top Foundation Wall: Insulation Distance To Top - The distance from the top of the foundation wall to the top of the foundation wall insulation. Only applies to basements/crawlspaces. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-foundation-walls'>HPXML Foundation Walls</a>) is used. + The distance from the top of the foundation wall to the top of the foundation wall insulation. Only applies to basements/crawlspaces. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-foundation-walls'>HPXML Foundation Walls</a>) is used. Double ft false @@ -1376,7 +1376,7 @@ foundation_wall_insulation_distance_to_bottom Foundation Wall: Insulation Distance To Bottom - The distance from the top of the foundation wall to the bottom of the foundation wall insulation. Only applies to basements/crawlspaces. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-foundation-walls'>HPXML Foundation Walls</a>) is used. + The distance from the top of the foundation wall to the bottom of the foundation wall insulation. Only applies to basements/crawlspaces. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-foundation-walls'>HPXML Foundation Walls</a>) is used. Double ft false @@ -1470,7 +1470,7 @@ slab_thickness Slab: Thickness - The thickness of the slab. Zero can be entered if there is a dirt floor instead of a slab. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-slabs'>HPXML Slabs</a>) is used. + The thickness of the slab. Zero can be entered if there is a dirt floor instead of a slab. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-slabs'>HPXML Slabs</a>) is used. Double in false @@ -1479,7 +1479,7 @@ slab_carpet_fraction Slab: Carpet Fraction - Fraction of the slab floor area that is carpeted. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-slabs'>HPXML Slabs</a>) is used. + Fraction of the slab floor area that is carpeted. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-slabs'>HPXML Slabs</a>) is used. Double Frac false @@ -1488,7 +1488,7 @@ slab_carpet_r Slab: Carpet R-value - R-value of the slab carpet. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-slabs'>HPXML Slabs</a>) is used. + R-value of the slab carpet. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-slabs'>HPXML Slabs</a>) is used. Double h-ft^2-R/Btu false @@ -1507,7 +1507,7 @@ roof_material_type Roof: Material Type - The material type of the roof. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-roofs'>HPXML Roofs</a>) is used. + The material type of the roof. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-roofs'>HPXML Roofs</a>) is used. Choice false false @@ -1553,7 +1553,7 @@ roof_color Roof: Color - The color of the roof. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-roofs'>HPXML Roofs</a>) is used. + The color of the roof. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-roofs'>HPXML Roofs</a>) is used. Choice false false @@ -1619,7 +1619,7 @@ radiant_barrier_grade Attic: Radiant Barrier Grade - The grade of the radiant barrier in the attic. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-roofs'>HPXML Roofs</a>) is used. + The grade of the radiant barrier in the attic. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-roofs'>HPXML Roofs</a>) is used. Choice false false @@ -1696,7 +1696,7 @@ wall_siding_type Wall: Siding Type - The siding type of the walls. Also applies to rim joists. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-walls'>HPXML Walls</a>) is used. + The siding type of the walls. Also applies to rim joists. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-walls'>HPXML Walls</a>) is used. Choice false false @@ -1750,7 +1750,7 @@ wall_color Wall: Color - The color of the walls. Also applies to rim joists. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-walls'>HPXML Walls</a>) is used. + The color of the walls. Also applies to rim joists. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-walls'>HPXML Walls</a>) is used. Choice false false @@ -1880,7 +1880,7 @@ window_fraction_operable Windows: Fraction Operable - Fraction of windows that are operable. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-windows'>HPXML Windows</a>) is used. + Fraction of windows that are operable. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-windows'>HPXML Windows</a>) is used. Double Frac false @@ -1889,7 +1889,7 @@ window_natvent_availability Windows: Natural Ventilation Availability - For operable windows, the number of days/week that windows can be opened by occupants for natural ventilation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-windows'>HPXML Windows</a>) is used. + For operable windows, the number of days/week that windows can be opened by occupants for natural ventilation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-windows'>HPXML Windows</a>) is used. Integer Days/week false @@ -1917,7 +1917,7 @@ window_interior_shading_type Windows: Interior Shading Type - Type of window interior shading. Summer/winter shading coefficients can be provided below instead. If neither is provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-interior-shading'>HPXML Interior Shading</a>) is used. + Type of window interior shading. Summer/winter shading coefficients can be provided below instead. If neither is provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-interior-shading'>HPXML Interior Shading</a>) is used. Choice false false @@ -1967,7 +1967,7 @@ window_interior_shading_winter Windows: Winter Interior Shading Coefficient - Interior shading coefficient for the winter season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-interior-shading'>HPXML Interior Shading</a>) is used. + Interior shading coefficient for the winter season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-interior-shading'>HPXML Interior Shading</a>) is used. Double Frac false @@ -1976,7 +1976,7 @@ window_interior_shading_summer Windows: Summer Interior Shading Coefficient - Interior shading coefficient for the summer season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-interior-shading'>HPXML Interior Shading</a>) is used. + Interior shading coefficient for the summer season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-interior-shading'>HPXML Interior Shading</a>) is used. Double Frac false @@ -1985,7 +1985,7 @@ window_exterior_shading_type Windows: Exterior Shading Type - Type of window exterior shading. Summer/winter shading coefficients can be provided below instead. If neither is provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-exterior-shading'>HPXML Exterior Shading</a>) is used. + Type of window exterior shading. Summer/winter shading coefficients can be provided below instead. If neither is provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-exterior-shading'>HPXML Exterior Shading</a>) is used. Choice false false @@ -2007,7 +2007,7 @@ window_exterior_shading_winter Windows: Winter Exterior Shading Coefficient - Exterior shading coefficient for the winter season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-exterior-shading'>HPXML Exterior Shading</a>) is used. + Exterior shading coefficient for the winter season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-exterior-shading'>HPXML Exterior Shading</a>) is used. Double Frac false @@ -2016,7 +2016,7 @@ window_exterior_shading_summer Windows: Summer Exterior Shading Coefficient - Exterior shading coefficient for the summer season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-exterior-shading'>HPXML Exterior Shading</a>) is used. + Exterior shading coefficient for the summer season, which if provided overrides the shading type input. 1.0 indicates no reduction in solar gain, 0.85 indicates 15% reduction, etc. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-exterior-shading'>HPXML Exterior Shading</a>) is used. Double Frac false @@ -2025,7 +2025,7 @@ window_shading_summer_season Windows: Shading Summer Season - Enter a date range like 'May 1 - Sep 30'. Defines the summer season for purposes of shading coefficients; the rest of the year is assumed to be winter. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-windows'>HPXML Windows</a>) is used. + Enter a date range like 'May 1 - Sep 30'. Defines the summer season for purposes of shading coefficients; the rest of the year is assumed to be winter. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-windows'>HPXML Windows</a>) is used. String false false @@ -2386,7 +2386,7 @@ air_leakage_has_flue_or_chimney_in_conditioned_space Air Leakage: Has Flue or Chimney in Conditioned Space - Presence of flue or chimney with combustion air from conditioned space; used for infiltration model. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#flue-or-chimney'>Flue or Chimney</a>) is used. + Presence of flue or chimney with combustion air from conditioned space; used for infiltration model. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#flue-or-chimney'>Flue or Chimney</a>) is used. Boolean false false @@ -2508,7 +2508,7 @@ heating_system_heating_capacity Heating System: Heating Capacity - The output heating capacity of the heating system. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-heating-systems'>HPXML Heating Systems</a>) is used. + The output heating capacity of the heating system. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-heating-systems'>HPXML Heating Systems</a>) is used. Double Btu/hr false @@ -2633,7 +2633,7 @@ cooling_system_cooling_compressor_type Cooling System: Cooling Compressor Type - The compressor type of the cooling system. Only applies to central air conditioner and mini-split. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#central-air-conditioner'>Central Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-air-conditioner'>Mini-Split Air Conditioner</a>) is used. + The compressor type of the cooling system. Only applies to central air conditioner and mini-split. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#central-air-conditioner'>Central Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-air-conditioner'>Mini-Split Air Conditioner</a>) is used. Choice false false @@ -2655,7 +2655,7 @@ cooling_system_cooling_sensible_heat_fraction Cooling System: Cooling Sensible Heat Fraction - The sensible heat fraction of the cooling system. Ignored for evaporative cooler. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#central-air-conditioner'>Central Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner'>Room Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-air-conditioner'>Packaged Terminal Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-air-conditioner'>Mini-Split Air Conditioner</a>) is used. + The sensible heat fraction of the cooling system. Ignored for evaporative cooler. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#central-air-conditioner'>Central Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner'>Room Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-air-conditioner'>Packaged Terminal Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-air-conditioner'>Mini-Split Air Conditioner</a>) is used. Double Frac false @@ -2664,7 +2664,7 @@ cooling_system_cooling_capacity Cooling System: Cooling Capacity - The output cooling capacity of the cooling system. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#central-air-conditioner'>Central Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner'>Room Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-air-conditioner'>Packaged Terminal Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#evaporative-cooler'>Evaporative Cooler</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-air-conditioner'>Mini-Split Air Conditioner</a>) is used. + The output cooling capacity of the cooling system. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#central-air-conditioner'>Central Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner'>Room Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-air-conditioner'>Packaged Terminal Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#evaporative-cooler'>Evaporative Cooler</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-air-conditioner'>Mini-Split Air Conditioner</a>) is used. Double Btu/hr false @@ -2737,7 +2737,7 @@ cooling_system_crankcase_heater_watts Cooling System: Crankcase Heater Power Watts - Cooling system crankcase heater power consumption in Watts. Applies only to central air conditioner, room air conditioner, packaged terminal air conditioner and mini-split. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#central-air-conditioner'>Central Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner'>Room Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-air-conditioner'>Packaged Terminal Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-air-conditioner'>Mini-Split Air Conditioner</a>) is used. + Cooling system crankcase heater power consumption in Watts. Applies only to central air conditioner, room air conditioner, packaged terminal air conditioner and mini-split. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#central-air-conditioner'>Central Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner'>Room Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-air-conditioner'>Packaged Terminal Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-air-conditioner'>Mini-Split Air Conditioner</a>) is used. Double W false @@ -2793,7 +2793,7 @@ cooling_system_integrated_heating_system_capacity Cooling System: Integrated Heating System Heating Capacity - The output heating capacity of the heating system integrated into cooling system. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner'>Room Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-air-conditioner'>Packaged Terminal Air Conditioner</a>) is used. Only used for room air conditioner and packaged terminal air conditioner. + The output heating capacity of the heating system integrated into cooling system. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner'>Room Air Conditioner</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-air-conditioner'>Packaged Terminal Air Conditioner</a>) is used. Only used for room air conditioner and packaged terminal air conditioner. Double Btu/hr false @@ -2914,7 +2914,7 @@ heat_pump_cooling_compressor_type Heat Pump: Cooling Compressor Type - The compressor type of the heat pump. Only applies to air-to-air and mini-split. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>) is used. + The compressor type of the heat pump. Only applies to air-to-air and mini-split. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>) is used. Choice false false @@ -2936,7 +2936,7 @@ heat_pump_cooling_sensible_heat_fraction Heat Pump: Cooling Sensible Heat Fraction - The sensible heat fraction of the heat pump. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#ground-to-air-heat-pump'>Ground-to-Air Heat Pump</a>) is used. + The sensible heat fraction of the heat pump. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#ground-to-air-heat-pump'>Ground-to-Air Heat Pump</a>) is used. Double Frac false @@ -2945,7 +2945,7 @@ heat_pump_heating_capacity Heat Pump: Heating Capacity - The output heating capacity of the heat pump. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#ground-to-air-heat-pump'>Ground-to-Air Heat Pump</a>) is used. + The output heating capacity of the heat pump. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#ground-to-air-heat-pump'>Ground-to-Air Heat Pump</a>) is used. Double Btu/hr false @@ -2971,7 +2971,7 @@ heat_pump_heating_capacity_retention_fraction Heat Pump: Heating Capacity Retention Fraction - The output heating capacity of the heat pump at a user-specified temperature (e.g., 17F or 5F) divided by the above nominal heating capacity. Applies to all heat pump types except ground-to-air. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>) is used. + The output heating capacity of the heat pump at a user-specified temperature (e.g., 17F or 5F) divided by the above nominal heating capacity. Applies to all heat pump types except ground-to-air. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>) is used. Double Frac false @@ -2989,7 +2989,7 @@ heat_pump_cooling_capacity Heat Pump: Cooling Capacity - The output cooling capacity of the heat pump. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#ground-to-air-heat-pump'>Ground-to-Air Heat Pump</a>) is used. + The output cooling capacity of the heat pump. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#ground-to-air-heat-pump'>Ground-to-Air Heat Pump</a>) is used. Double Btu/hr false @@ -3035,7 +3035,7 @@ heat_pump_compressor_lockout_temp Heat Pump: Compressor Lockout Temperature - The temperature below which the heat pump compressor is disabled. If both this and Backup Heating Lockout Temperature are provided and use the same value, it essentially defines a switchover temperature (for, e.g., a dual-fuel heat pump). Applies to all heat pump types other than ground-to-air. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>) is used. + The temperature below which the heat pump compressor is disabled. If both this and Backup Heating Lockout Temperature are provided and use the same value, it essentially defines a switchover temperature (for, e.g., a dual-fuel heat pump). Applies to all heat pump types other than ground-to-air. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>) is used. Double F false @@ -3120,7 +3120,7 @@ heat_pump_backup_heating_capacity Heat Pump: Backup Heating Capacity - The backup output heating capacity of the heat pump. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#backup'>Backup</a>) is used. Only applies if Backup Type is 'integrated'. + The backup output heating capacity of the heat pump. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#backup'>Backup</a>) is used. Only applies if Backup Type is 'integrated'. Double Btu/hr false @@ -3129,7 +3129,7 @@ heat_pump_backup_heating_lockout_temp Heat Pump: Backup Heating Lockout Temperature - The temperature above which the heat pump backup system is disabled. If both this and Compressor Lockout Temperature are provided and use the same value, it essentially defines a switchover temperature (for, e.g., a dual-fuel heat pump). Applies for both Backup Type of 'integrated' and 'separate'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#backup'>Backup</a>) is used. + The temperature above which the heat pump backup system is disabled. If both this and Compressor Lockout Temperature are provided and use the same value, it essentially defines a switchover temperature (for, e.g., a dual-fuel heat pump). Applies for both Backup Type of 'integrated' and 'separate'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#backup'>Backup</a>) is used. Double F false @@ -3138,7 +3138,7 @@ heat_pump_sizing_methodology Heat Pump: Sizing Methodology - The auto-sizing methodology to use when the heat pump capacity is not provided. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-hvac-sizing-control'>HPXML HVAC Sizing Control</a>) is used. + The auto-sizing methodology to use when the heat pump capacity is not provided. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-hvac-sizing-control'>HPXML HVAC Sizing Control</a>) is used. Choice false false @@ -3160,7 +3160,7 @@ heat_pump_backup_sizing_methodology Heat Pump: Backup Sizing Methodology - The auto-sizing methodology to use when the heat pump backup capacity is not provided. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-hvac-sizing-control'>HPXML HVAC Sizing Control</a>) is used. + The auto-sizing methodology to use when the heat pump backup capacity is not provided. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-hvac-sizing-control'>HPXML HVAC Sizing Control</a>) is used. Choice false false @@ -3214,7 +3214,7 @@ heat_pump_crankcase_heater_watts Heat Pump: Crankcase Heater Power Watts - Heat Pump crankcase heater power consumption in Watts. Applies only to air-to-air, mini-split, packaged terminal heat pump and room air conditioner with reverse cycle. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>) is used. + Heat Pump crankcase heater power consumption in Watts. Applies only to air-to-air, mini-split, packaged terminal heat pump and room air conditioner with reverse cycle. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-to-air-heat-pump'>Air-to-Air Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#mini-split-heat-pump'>Mini-Split Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#packaged-terminal-heat-pump'>Packaged Terminal Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#room-air-conditioner-w-reverse-cycle'>Room Air Conditioner w/ Reverse Cycle</a>) is used. Double W false @@ -3332,7 +3332,7 @@ geothermal_loop_configuration Geothermal Loop: Configuration - Configuration of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#ground-to-air-heat-pump'>Ground-to-Air Heat Pump</a>) is used. + Configuration of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#ground-to-air-heat-pump'>Ground-to-Air Heat Pump</a>) is used. Choice false false @@ -3350,7 +3350,7 @@ geothermal_loop_borefield_configuration Geothermal Loop: Borefield Configuration - Borefield configuration of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. + Borefield configuration of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. Choice false false @@ -3384,7 +3384,7 @@ geothermal_loop_loop_flow Geothermal Loop: Loop Flow - Water flow rate through the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. + Water flow rate through the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. Double gpm false @@ -3393,7 +3393,7 @@ geothermal_loop_boreholes_count Geothermal Loop: Boreholes Count - Number of boreholes. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. + Number of boreholes. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. Integer # false @@ -3402,7 +3402,7 @@ geothermal_loop_boreholes_length Geothermal Loop: Boreholes Length - Average length of each borehole (vertical). Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. + Average length of each borehole (vertical). Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. Double ft false @@ -3411,7 +3411,7 @@ geothermal_loop_boreholes_spacing Geothermal Loop: Boreholes Spacing - Distance between bores. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. + Distance between bores. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. Double ft false @@ -3420,7 +3420,7 @@ geothermal_loop_boreholes_diameter Geothermal Loop: Boreholes Diameter - Diameter of bores. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. + Diameter of bores. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. Double in false @@ -3429,7 +3429,7 @@ geothermal_loop_grout_type Geothermal Loop: Grout Type - Grout type of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. + Grout type of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. Choice false false @@ -3447,7 +3447,7 @@ geothermal_loop_pipe_type Geothermal Loop: Pipe Type - Pipe type of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. + Pipe type of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. Choice false false @@ -3465,7 +3465,7 @@ geothermal_loop_pipe_diameter Geothermal Loop: Pipe Diameter - Pipe diameter of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. + Pipe diameter of the geothermal loop. Only applies to ground-to-air heat pump type. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-geothermal-loops'>HPXML Geothermal Loops</a>) is used. Choice in false @@ -3584,7 +3584,7 @@ heating_system_2_heating_capacity Heating System 2: Heating Capacity - The output heating capacity of the second heating system. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-heating-systems'>HPXML Heating Systems</a>) is used. + The output heating capacity of the second heating system. If not provided, the OS-HPXML autosized default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-heating-systems'>HPXML Heating Systems</a>) is used. Double Btu/hr false @@ -3656,7 +3656,7 @@ hvac_control_heating_season_period HVAC Control: Heating Season Period - Enter a date range like 'Nov 1 - Jun 30'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-hvac-control'>HPXML HVAC Control</a>) is used. Can also provide 'BuildingAmerica' to use automatic seasons from the Building America House Simulation Protocols. + Enter a date range like 'Nov 1 - Jun 30'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-hvac-control'>HPXML HVAC Control</a>) is used. Can also provide 'BuildingAmerica' to use automatic seasons from the Building America House Simulation Protocols. String false false @@ -3664,7 +3664,7 @@ hvac_control_cooling_season_period HVAC Control: Cooling Season Period - Enter a date range like 'Jun 1 - Oct 31'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-hvac-control'>HPXML HVAC Control</a>) is used. Can also provide 'BuildingAmerica' to use automatic seasons from the Building America House Simulation Protocols. + Enter a date range like 'Jun 1 - Oct 31'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-hvac-control'>HPXML HVAC Control</a>) is used. Can also provide 'BuildingAmerica' to use automatic seasons from the Building America House Simulation Protocols. String false false @@ -3672,7 +3672,7 @@ hvac_blower_fan_watts_per_cfm HVAC Blower: Fan Efficiency - The blower fan efficiency at maximum fan speed. Applies only to split (not packaged) systems (i.e., applies to ducted systems as well as ductless mini-split systems). If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-heating-systems'>HPXML Heating Systems</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-cooling-systems'>HPXML Cooling Systems</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-heat-pumps'>HPXML Heat Pumps</a>) is used. + The blower fan efficiency at maximum fan speed. Applies only to split (not packaged) systems (i.e., applies to ducted systems as well as ductless mini-split systems). If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-heating-systems'>HPXML Heating Systems</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-cooling-systems'>HPXML Cooling Systems</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-heat-pumps'>HPXML Heat Pumps</a>) is used. Double W/CFM false @@ -3713,7 +3713,7 @@ ducts_supply_location Ducts: Supply Location - The location of the supply ducts. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. + The location of the supply ducts. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. Choice false false @@ -3839,7 +3839,7 @@ ducts_supply_surface_area Ducts: Supply Surface Area - The supply ducts surface area in the given location. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. + The supply ducts surface area in the given location. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. Double ft^2 false @@ -3848,7 +3848,7 @@ ducts_supply_surface_area_fraction Ducts: Supply Area Fraction - The fraction of supply ducts surface area in the given location. Only used if Surface Area is not provided. If the fraction is less than 1, the remaining duct area is assumed to be in conditioned space. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. + The fraction of supply ducts surface area in the given location. Only used if Surface Area is not provided. If the fraction is less than 1, the remaining duct area is assumed to be in conditioned space. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. Double frac false @@ -3857,7 +3857,7 @@ ducts_supply_fraction_rectangular Ducts: Supply Fraction Rectangular - The fraction of supply ducts that are rectangular (as opposed to round); this affects the duct effective R-value used for modeling. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. + The fraction of supply ducts that are rectangular (as opposed to round); this affects the duct effective R-value used for modeling. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. Double frac false @@ -3875,7 +3875,7 @@ ducts_return_location Ducts: Return Location - The location of the return ducts. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. + The location of the return ducts. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. Choice false false @@ -4001,7 +4001,7 @@ ducts_return_surface_area Ducts: Return Surface Area - The return ducts surface area in the given location. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. + The return ducts surface area in the given location. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. Double ft^2 false @@ -4010,7 +4010,7 @@ ducts_return_surface_area_fraction Ducts: Return Area Fraction - The fraction of return ducts surface area in the given location. Only used if Surface Area is not provided. If the fraction is less than 1, the remaining duct area is assumed to be in conditioned space. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. + The fraction of return ducts surface area in the given location. Only used if Surface Area is not provided. If the fraction is less than 1, the remaining duct area is assumed to be in conditioned space. If neither Surface Area nor Area Fraction provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. Double frac false @@ -4019,7 +4019,7 @@ ducts_number_of_return_registers Ducts: Number of Return Registers - The number of return registers of the ducts. Only used to calculate default return duct surface area. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. + The number of return registers of the ducts. Only used to calculate default return duct surface area. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. Integer # false @@ -4028,7 +4028,7 @@ ducts_return_fraction_rectangular Ducts: Return Fraction Rectangular - The fraction of return ducts that are rectangular (as opposed to round); this affects the duct effective R-value used for modeling. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. + The fraction of return ducts that are rectangular (as opposed to round); this affects the duct effective R-value used for modeling. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#air-distribution'>Air Distribution</a>) is used. Double frac false @@ -4076,7 +4076,7 @@ mech_vent_flow_rate Mechanical Ventilation: Flow Rate - The flow rate of the mechanical ventilation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-mechanical-ventilation-fans'>HPXML Mechanical Ventilation Fans</a>) is used. + The flow rate of the mechanical ventilation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-mechanical-ventilation-fans'>HPXML Mechanical Ventilation Fans</a>) is used. Double CFM false @@ -4085,7 +4085,7 @@ mech_vent_hours_in_operation Mechanical Ventilation: Hours In Operation - The hours in operation of the mechanical ventilation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-mechanical-ventilation-fans'>HPXML Mechanical Ventilation Fans</a>) is used. + The hours in operation of the mechanical ventilation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-mechanical-ventilation-fans'>HPXML Mechanical Ventilation Fans</a>) is used. Double hrs/day false @@ -4133,7 +4133,7 @@ mech_vent_fan_power Mechanical Ventilation: Fan Power - The fan power of the mechanical ventilation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-mechanical-ventilation-fans'>HPXML Mechanical Ventilation Fans</a>) is used. + The fan power of the mechanical ventilation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-mechanical-ventilation-fans'>HPXML Mechanical Ventilation Fans</a>) is used. Double W false @@ -4353,7 +4353,7 @@ kitchen_fans_quantity Kitchen Fans: Quantity - The quantity of the kitchen fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The quantity of the kitchen fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Integer # false @@ -4362,7 +4362,7 @@ kitchen_fans_flow_rate Kitchen Fans: Flow Rate - The flow rate of the kitchen fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The flow rate of the kitchen fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Double CFM false @@ -4371,7 +4371,7 @@ kitchen_fans_hours_in_operation Kitchen Fans: Hours In Operation - The hours in operation of the kitchen fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The hours in operation of the kitchen fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Double hrs/day false @@ -4380,7 +4380,7 @@ kitchen_fans_power Kitchen Fans: Fan Power - The fan power of the kitchen fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The fan power of the kitchen fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Double W false @@ -4389,7 +4389,7 @@ kitchen_fans_start_hour Kitchen Fans: Start Hour - The start hour of the kitchen fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The start hour of the kitchen fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Integer hr false @@ -4398,7 +4398,7 @@ bathroom_fans_quantity Bathroom Fans: Quantity - The quantity of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The quantity of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Integer # false @@ -4407,7 +4407,7 @@ bathroom_fans_flow_rate Bathroom Fans: Flow Rate - The flow rate of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The flow rate of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Double CFM false @@ -4416,7 +4416,7 @@ bathroom_fans_hours_in_operation Bathroom Fans: Hours In Operation - The hours in operation of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The hours in operation of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Double hrs/day false @@ -4425,7 +4425,7 @@ bathroom_fans_power Bathroom Fans: Fan Power - The fan power of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The fan power of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Double W false @@ -4434,7 +4434,7 @@ bathroom_fans_start_hour Bathroom Fans: Start Hour - The start hour of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. + The start hour of the bathroom fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-local-ventilation-fans'>HPXML Local Ventilation Fans</a>) is used. Integer hr false @@ -4462,7 +4462,7 @@ whole_house_fan_flow_rate Whole House Fan: Flow Rate - The flow rate of the whole house fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-whole-house-fans'>HPXML Whole House Fans</a>) is used. + The flow rate of the whole house fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-whole-house-fans'>HPXML Whole House Fans</a>) is used. Double CFM false @@ -4471,7 +4471,7 @@ whole_house_fan_power Whole House Fan: Fan Power - The fan power of the whole house fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-whole-house-fans'>HPXML Whole House Fans</a>) is used. + The fan power of the whole house fan. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-whole-house-fans'>HPXML Whole House Fans</a>) is used. Double W false @@ -4550,7 +4550,7 @@ water_heater_location Water Heater: Location - The location of water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-water-heating-systems'>HPXML Water Heating Systems</a>) is used. + The location of water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-water-heating-systems'>HPXML Water Heating Systems</a>) is used. Choice false false @@ -4624,7 +4624,7 @@ water_heater_tank_volume Water Heater: Tank Volume - Nominal volume of water heater tank. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#conventional-storage'>Conventional Storage</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#heat-pump'>Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#combi-boiler-w-storage'>Combi Boiler w/ Storage</a>) is used. + Nominal volume of water heater tank. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#conventional-storage'>Conventional Storage</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#heat-pump'>Heat Pump</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#combi-boiler-w-storage'>Combi Boiler w/ Storage</a>) is used. Double gal false @@ -4661,7 +4661,7 @@ water_heater_usage_bin Water Heater: Usage Bin - The usage of the water heater. Only applies if Efficiency Type is UniformEnergyFactor and Type is not instantaneous water heater. Does not apply to space-heating boilers. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#conventional-storage'>Conventional Storage</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#heat-pump'>Heat Pump</a>) is used. + The usage of the water heater. Only applies if Efficiency Type is UniformEnergyFactor and Type is not instantaneous water heater. Does not apply to space-heating boilers. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#conventional-storage'>Conventional Storage</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#heat-pump'>Heat Pump</a>) is used. Choice false false @@ -4687,7 +4687,7 @@ water_heater_recovery_efficiency Water Heater: Recovery Efficiency - Ratio of energy delivered to water heater to the energy content of the fuel consumed by the water heater. Only used for non-electric storage water heaters. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#conventional-storage'>Conventional Storage</a>) is used. + Ratio of energy delivered to water heater to the energy content of the fuel consumed by the water heater. Only used for non-electric storage water heaters. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#conventional-storage'>Conventional Storage</a>) is used. Double Frac false @@ -4696,7 +4696,7 @@ water_heater_heating_capacity Water Heater: Heating Capacity - Heating capacity. Only applies to storage water heater and heat pump water heater (compressor). If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#conventional-storage'>Conventional Storage</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#heat-pump'>Heat Pump</a>) is used. + Heating capacity. Only applies to storage water heater and heat pump water heater (compressor). If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#conventional-storage'>Conventional Storage</a>, <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#heat-pump'>Heat Pump</a>) is used. Double Btu/hr false @@ -4705,7 +4705,7 @@ water_heater_backup_heating_capacity Water Heater: Backup Heating Capacity - Backup heating capacity for a heat pump water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#heat-pump'>Heat Pump</a>) is used. + Backup heating capacity for a heat pump water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#heat-pump'>Heat Pump</a>) is used. Double Btu/hr false @@ -4714,7 +4714,7 @@ water_heater_standby_loss Water Heater: Standby Loss - The standby loss of water heater. Only applies to space-heating boilers. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#combi-boiler-w-storage'>Combi Boiler w/ Storage</a>) is used. + The standby loss of water heater. Only applies to space-heating boilers. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#combi-boiler-w-storage'>Combi Boiler w/ Storage</a>) is used. Double F/hr false @@ -4732,7 +4732,7 @@ water_heater_setpoint_temperature Water Heater: Setpoint Temperature - The setpoint temperature of water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-water-heating-systems'>HPXML Water Heating Systems</a>) is used. + The setpoint temperature of water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-water-heating-systems'>HPXML Water Heating Systems</a>) is used. Double F false @@ -4768,7 +4768,7 @@ water_heater_tank_model_type Water Heater: Tank Type - Type of tank model to use. The 'stratified' tank generally provide more accurate results, but may significantly increase run time. Applies only to storage water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#conventional-storage'>Conventional Storage</a>) is used. + Type of tank model to use. The 'stratified' tank generally provide more accurate results, but may significantly increase run time. Applies only to storage water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#conventional-storage'>Conventional Storage</a>) is used. Choice false false @@ -4786,7 +4786,7 @@ water_heater_operating_mode Water Heater: Operating Mode - The water heater operating mode. The 'heat pump only' option only uses the heat pump, while 'hybrid/auto' allows the backup electric resistance to come on in high demand situations. This is ignored if a scheduled operating mode type is selected. Applies only to heat pump water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#heat-pump'>Heat Pump</a>) is used. + The water heater operating mode. The 'heat pump only' option only uses the heat pump, while 'hybrid/auto' allows the backup electric resistance to come on in high demand situations. This is ignored if a scheduled operating mode type is selected. Applies only to heat pump water heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#heat-pump'>Heat Pump</a>) is used. Choice false false @@ -4823,7 +4823,7 @@ hot_water_distribution_standard_piping_length Hot Water Distribution: Standard Piping Length - If the distribution system is Standard, the length of the piping. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#standard'>Standard</a>) is used. + If the distribution system is Standard, the length of the piping. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#standard'>Standard</a>) is used. Double ft false @@ -4863,7 +4863,7 @@ hot_water_distribution_recirc_piping_length Hot Water Distribution: Recirculation Piping Length - If the distribution system is Recirculation, the length of the recirculation piping. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#recirculation-in-unit'>Recirculation (In-Unit)</a>) is used. + If the distribution system is Recirculation, the length of the recirculation piping. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#recirculation-in-unit'>Recirculation (In-Unit)</a>) is used. Double ft false @@ -4872,7 +4872,7 @@ hot_water_distribution_recirc_branch_piping_length Hot Water Distribution: Recirculation Branch Piping Length - If the distribution system is Recirculation, the length of the recirculation branch piping. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#recirculation-in-unit'>Recirculation (In-Unit)</a>) is used. + If the distribution system is Recirculation, the length of the recirculation branch piping. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#recirculation-in-unit'>Recirculation (In-Unit)</a>) is used. Double ft false @@ -4881,7 +4881,7 @@ hot_water_distribution_recirc_pump_power Hot Water Distribution: Recirculation Pump Power - If the distribution system is Recirculation, the recirculation pump power. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#recirculation-in-unit'>Recirculation (In-Unit)</a>) is used. + If the distribution system is Recirculation, the recirculation pump power. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#recirculation-in-unit'>Recirculation (In-Unit)</a>) is used. Double W false @@ -4890,7 +4890,7 @@ hot_water_distribution_pipe_r Hot Water Distribution: Pipe Insulation Nominal R-Value - Nominal R-value of the pipe insulation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-hot-water-distribution'>HPXML Hot Water Distribution</a>) is used. + Nominal R-value of the pipe insulation. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-hot-water-distribution'>HPXML Hot Water Distribution</a>) is used. Double h-ft^2-R/Btu false @@ -4989,7 +4989,7 @@ water_fixtures_usage_multiplier Hot Water Fixtures: Usage Multiplier - Multiplier on the hot water usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-water-fixtures'>HPXML Water Fixtures</a>) is used. + Multiplier on the hot water usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-water-fixtures'>HPXML Water Fixtures</a>) is used. Double false false @@ -4997,7 +4997,7 @@ general_water_use_usage_multiplier General Water Use: Usage Multiplier - Multiplier on internal gains from general water use (floor mopping, shower evaporation, water films on showers, tubs & sinks surfaces, plant watering, etc.) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-building-occupancy'>HPXML Building Occupancy</a>) is used. + Multiplier on internal gains from general water use (floor mopping, shower evaporation, water films on showers, tubs & sinks surfaces, plant watering, etc.) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-building-occupancy'>HPXML Building Occupancy</a>) is used. Double false false @@ -5124,7 +5124,7 @@ solar_thermal_storage_volume Solar Thermal: Storage Volume - The storage volume of the solar thermal system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#detailed-inputs'>Detailed Inputs</a>) is used. + The storage volume of the solar thermal system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#detailed-inputs'>Detailed Inputs</a>) is used. Double gal false @@ -5162,7 +5162,7 @@ pv_system_module_type PV System: Module Type - Module type of the PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. + Module type of the PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. Choice false false @@ -5184,7 +5184,7 @@ pv_system_location PV System: Location - Location of the PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. + Location of the PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. Choice false false @@ -5202,7 +5202,7 @@ pv_system_tracking PV System: Tracking - Type of tracking for the PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. + Type of tracking for the PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. Choice false false @@ -5258,7 +5258,7 @@ pv_system_inverter_efficiency PV System: Inverter Efficiency - Inverter efficiency of the PV system. If there are two PV systems, this will apply to both. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. + Inverter efficiency of the PV system. If there are two PV systems, this will apply to both. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. Double Frac false @@ -5267,7 +5267,7 @@ pv_system_system_losses_fraction PV System: System Losses Fraction - System losses fraction of the PV system. If there are two PV systems, this will apply to both. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. + System losses fraction of the PV system. If there are two PV systems, this will apply to both. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. Double Frac false @@ -5304,7 +5304,7 @@ pv_system_2_module_type PV System 2: Module Type - Module type of the second PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. + Module type of the second PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. Choice false false @@ -5326,7 +5326,7 @@ pv_system_2_location PV System 2: Location - Location of the second PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. + Location of the second PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. Choice false false @@ -5344,7 +5344,7 @@ pv_system_2_tracking PV System 2: Tracking - Type of tracking for the second PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. + Type of tracking for the second PV system. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-photovoltaics'>HPXML Photovoltaics</a>) is used. Choice false false @@ -5419,7 +5419,7 @@ battery_location Battery: Location - The space type for the lithium ion battery location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. + The space type for the lithium ion battery location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. Choice false false @@ -5477,7 +5477,7 @@ battery_power Battery: Rated Power Output - The rated power output of the lithium ion battery. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. + The rated power output of the lithium ion battery. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. Double W false @@ -5486,7 +5486,7 @@ battery_capacity Battery: Nominal Capacity - The nominal capacity of the lithium ion battery. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. + The nominal capacity of the lithium ion battery. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. Double kWh false @@ -5495,7 +5495,7 @@ battery_usable_capacity Battery: Usable Capacity - The usable capacity of the lithium ion battery. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. + The usable capacity of the lithium ion battery. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. Double kWh false @@ -5504,7 +5504,7 @@ battery_round_trip_efficiency Battery: Round Trip Efficiency - The round trip efficiency of the lithium ion battery. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. + The round trip efficiency of the lithium ion battery. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-batteries'>HPXML Batteries</a>) is used. Double Frac false @@ -5568,7 +5568,7 @@ lighting_interior_usage_multiplier Lighting: Interior Usage Multiplier - Multiplier on the lighting energy usage (interior) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. + Multiplier on the lighting energy usage (interior) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. Double false false @@ -5603,7 +5603,7 @@ lighting_exterior_usage_multiplier Lighting: Exterior Usage Multiplier - Multiplier on the lighting energy usage (exterior) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. + Multiplier on the lighting energy usage (exterior) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. Double false false @@ -5638,7 +5638,7 @@ lighting_garage_usage_multiplier Lighting: Garage Usage Multiplier - Multiplier on the lighting energy usage (garage) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. + Multiplier on the lighting energy usage (garage) that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. Double false false @@ -5665,7 +5665,7 @@ holiday_lighting_daily_kwh Holiday Lighting: Daily Consumption - The daily energy consumption for holiday lighting (exterior). If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. + The daily energy consumption for holiday lighting (exterior). If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. Double kWh/day false @@ -5674,7 +5674,7 @@ holiday_lighting_period Holiday Lighting: Period - Enter a date range like 'Nov 25 - Jan 5'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. + Enter a date range like 'Nov 25 - Jan 5'. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-lighting'>HPXML Lighting</a>) is used. String false false @@ -5783,7 +5783,7 @@ clothes_washer_location Clothes Washer: Location - The space type for the clothes washer location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. + The space type for the clothes washer location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. Choice false false @@ -5844,7 +5844,7 @@ clothes_washer_efficiency Clothes Washer: Efficiency - The efficiency of the clothes washer. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. + The efficiency of the clothes washer. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. Double ft^3/kWh-cyc false @@ -5853,7 +5853,7 @@ clothes_washer_rated_annual_kwh Clothes Washer: Rated Annual Consumption - The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. + The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. Double kWh/yr false @@ -5862,7 +5862,7 @@ clothes_washer_label_electric_rate Clothes Washer: Label Electric Rate - The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. + The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. Double $/kWh false @@ -5871,7 +5871,7 @@ clothes_washer_label_gas_rate Clothes Washer: Label Gas Rate - The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. + The annual energy consumed by the clothes washer, as rated, obtained from the EnergyGuide label. This includes both the appliance electricity consumption and the energy required for water heating. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. Double $/therm false @@ -5880,7 +5880,7 @@ clothes_washer_label_annual_gas_cost Clothes Washer: Label Annual Cost with Gas DHW - The annual cost of using the system under test conditions. Input is obtained from the EnergyGuide label. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. + The annual cost of using the system under test conditions. Input is obtained from the EnergyGuide label. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. Double $ false @@ -5889,7 +5889,7 @@ clothes_washer_label_usage Clothes Washer: Label Usage - The clothes washer loads per week. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. + The clothes washer loads per week. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. Double cyc/wk false @@ -5898,7 +5898,7 @@ clothes_washer_capacity Clothes Washer: Drum Volume - Volume of the washer drum. Obtained from the EnergyStar website or the manufacturer's literature. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. + Volume of the washer drum. Obtained from the EnergyStar website or the manufacturer's literature. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. Double ft^3 false @@ -5907,7 +5907,7 @@ clothes_washer_usage_multiplier Clothes Washer: Usage Multiplier - Multiplier on the clothes washer energy and hot water usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. + Multiplier on the clothes washer energy and hot water usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-washer'>HPXML Clothes Washer</a>) is used. Double false false @@ -5934,7 +5934,7 @@ clothes_dryer_location Clothes Dryer: Location - The space type for the clothes dryer location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-dryer'>HPXML Clothes Dryer</a>) is used. + The space type for the clothes dryer location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-dryer'>HPXML Clothes Dryer</a>) is used. Choice false false @@ -6030,7 +6030,7 @@ clothes_dryer_efficiency Clothes Dryer: Efficiency - The efficiency of the clothes dryer. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-dryer'>HPXML Clothes Dryer</a>) is used. + The efficiency of the clothes dryer. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-dryer'>HPXML Clothes Dryer</a>) is used. Double lb/kWh false @@ -6039,7 +6039,7 @@ clothes_dryer_vented_flow_rate Clothes Dryer: Vented Flow Rate - The exhaust flow rate of the vented clothes dryer. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-dryer'>HPXML Clothes Dryer</a>) is used. + The exhaust flow rate of the vented clothes dryer. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-dryer'>HPXML Clothes Dryer</a>) is used. Double CFM false @@ -6048,7 +6048,7 @@ clothes_dryer_usage_multiplier Clothes Dryer: Usage Multiplier - Multiplier on the clothes dryer energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-clothes-dryer'>HPXML Clothes Dryer</a>) is used. + Multiplier on the clothes dryer energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-clothes-dryer'>HPXML Clothes Dryer</a>) is used. Double false false @@ -6075,7 +6075,7 @@ dishwasher_location Dishwasher: Location - The space type for the dishwasher location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. + The space type for the dishwasher location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. Choice false false @@ -6136,7 +6136,7 @@ dishwasher_efficiency Dishwasher: Efficiency - The efficiency of the dishwasher. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. + The efficiency of the dishwasher. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. Double RatedAnnualkWh or EnergyFactor false @@ -6145,7 +6145,7 @@ dishwasher_label_electric_rate Dishwasher: Label Electric Rate - The label electric rate of the dishwasher. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. + The label electric rate of the dishwasher. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. Double $/kWh false @@ -6154,7 +6154,7 @@ dishwasher_label_gas_rate Dishwasher: Label Gas Rate - The label gas rate of the dishwasher. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. + The label gas rate of the dishwasher. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. Double $/therm false @@ -6163,7 +6163,7 @@ dishwasher_label_annual_gas_cost Dishwasher: Label Annual Gas Cost - The label annual gas cost of the dishwasher. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. + The label annual gas cost of the dishwasher. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. Double $ false @@ -6172,7 +6172,7 @@ dishwasher_label_usage Dishwasher: Label Usage - The dishwasher loads per week. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. + The dishwasher loads per week. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. Double cyc/wk false @@ -6181,7 +6181,7 @@ dishwasher_place_setting_capacity Dishwasher: Number of Place Settings - The number of place settings for the unit. Data obtained from manufacturer's literature. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. + The number of place settings for the unit. Data obtained from manufacturer's literature. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. Integer # false @@ -6190,7 +6190,7 @@ dishwasher_usage_multiplier Dishwasher: Usage Multiplier - Multiplier on the dishwasher energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. + Multiplier on the dishwasher energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-dishwasher'>HPXML Dishwasher</a>) is used. Double false false @@ -6217,7 +6217,7 @@ refrigerator_location Refrigerator: Location - The space type for the refrigerator location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. + The space type for the refrigerator location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. Choice false false @@ -6259,7 +6259,7 @@ refrigerator_rated_annual_kwh Refrigerator: Rated Annual Consumption - The EnergyGuide rated annual energy consumption for a refrigerator. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. + The EnergyGuide rated annual energy consumption for a refrigerator. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. Double kWh/yr false @@ -6268,7 +6268,7 @@ refrigerator_usage_multiplier Refrigerator: Usage Multiplier - Multiplier on the refrigerator energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. + Multiplier on the refrigerator energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. Double false false @@ -6295,7 +6295,7 @@ extra_refrigerator_location Extra Refrigerator: Location - The space type for the extra refrigerator location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. + The space type for the extra refrigerator location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. Choice false false @@ -6337,7 +6337,7 @@ extra_refrigerator_rated_annual_kwh Extra Refrigerator: Rated Annual Consumption - The EnergyGuide rated annual energy consumption for an extra refrigerator. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. + The EnergyGuide rated annual energy consumption for an extra refrigerator. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. Double kWh/yr false @@ -6346,7 +6346,7 @@ extra_refrigerator_usage_multiplier Extra Refrigerator: Usage Multiplier - Multiplier on the extra refrigerator energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. + Multiplier on the extra refrigerator energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-refrigerators'>HPXML Refrigerators</a>) is used. Double false false @@ -6373,7 +6373,7 @@ freezer_location Freezer: Location - The space type for the freezer location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-freezers'>HPXML Freezers</a>) is used. + The space type for the freezer location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-freezers'>HPXML Freezers</a>) is used. Choice false false @@ -6415,7 +6415,7 @@ freezer_rated_annual_kwh Freezer: Rated Annual Consumption - The EnergyGuide rated annual energy consumption for a freezer. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-freezers'>HPXML Freezers</a>) is used. + The EnergyGuide rated annual energy consumption for a freezer. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-freezers'>HPXML Freezers</a>) is used. Double kWh/yr false @@ -6424,7 +6424,7 @@ freezer_usage_multiplier Freezer: Usage Multiplier - Multiplier on the freezer energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-freezers'>HPXML Freezers</a>) is used. + Multiplier on the freezer energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-freezers'>HPXML Freezers</a>) is used. Double false false @@ -6451,7 +6451,7 @@ cooking_range_oven_location Cooking Range/Oven: Location - The space type for the cooking range/oven location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-cooking-range-oven'>HPXML Cooking Range/Oven</a>) is used. + The space type for the cooking range/oven location. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-cooking-range-oven'>HPXML Cooking Range/Oven</a>) is used. Choice false false @@ -6528,7 +6528,7 @@ cooking_range_oven_is_induction Cooking Range/Oven: Is Induction - Whether the cooking range is induction. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-cooking-range-oven'>HPXML Cooking Range/Oven</a>) is used. + Whether the cooking range is induction. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-cooking-range-oven'>HPXML Cooking Range/Oven</a>) is used. Boolean false false @@ -6546,7 +6546,7 @@ cooking_range_oven_is_convection Cooking Range/Oven: Is Convection - Whether the oven is convection. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-cooking-range-oven'>HPXML Cooking Range/Oven</a>) is used. + Whether the oven is convection. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-cooking-range-oven'>HPXML Cooking Range/Oven</a>) is used. Boolean false false @@ -6564,7 +6564,7 @@ cooking_range_oven_usage_multiplier Cooking Range/Oven: Usage Multiplier - Multiplier on the cooking range/oven energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-cooking-range-oven'>HPXML Cooking Range/Oven</a>) is used. + Multiplier on the cooking range/oven energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-cooking-range-oven'>HPXML Cooking Range/Oven</a>) is used. Double false false @@ -6591,7 +6591,7 @@ ceiling_fan_label_energy_use Ceiling Fan: Label Energy Use - The label average energy use of the ceiling fan(s). If neither Efficiency nor Label Energy Use provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-ceiling-fans'>HPXML Ceiling Fans</a>) is used. + The label average energy use of the ceiling fan(s). If neither Efficiency nor Label Energy Use provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-ceiling-fans'>HPXML Ceiling Fans</a>) is used. Double W false @@ -6600,7 +6600,7 @@ ceiling_fan_efficiency Ceiling Fan: Efficiency - The efficiency rating of the ceiling fan(s) at medium speed. Only used if Label Energy Use not provided. If neither Efficiency nor Label Energy Use provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-ceiling-fans'>HPXML Ceiling Fans</a>) is used. + The efficiency rating of the ceiling fan(s) at medium speed. Only used if Label Energy Use not provided. If neither Efficiency nor Label Energy Use provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-ceiling-fans'>HPXML Ceiling Fans</a>) is used. Double CFM/W false @@ -6609,7 +6609,7 @@ ceiling_fan_quantity Ceiling Fan: Quantity - Total number of ceiling fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-ceiling-fans'>HPXML Ceiling Fans</a>) is used. + Total number of ceiling fans. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-ceiling-fans'>HPXML Ceiling Fans</a>) is used. Integer # false @@ -6618,7 +6618,7 @@ ceiling_fan_cooling_setpoint_temp_offset Ceiling Fan: Cooling Setpoint Temperature Offset - The cooling setpoint temperature offset during months when the ceiling fans are operating. Only applies if ceiling fan quantity is greater than zero. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-ceiling-fans'>HPXML Ceiling Fans</a>) is used. + The cooling setpoint temperature offset during months when the ceiling fans are operating. Only applies if ceiling fan quantity is greater than zero. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-ceiling-fans'>HPXML Ceiling Fans</a>) is used. Double F false @@ -6646,7 +6646,7 @@ misc_plug_loads_television_annual_kwh Misc Plug Loads: Television Annual kWh - The annual energy consumption of the television plug loads. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + The annual energy consumption of the television plug loads. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double kWh/yr false @@ -6655,7 +6655,7 @@ misc_plug_loads_television_usage_multiplier Misc Plug Loads: Television Usage Multiplier - Multiplier on the television energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + Multiplier on the television energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double false false @@ -6663,7 +6663,7 @@ misc_plug_loads_other_annual_kwh Misc Plug Loads: Other Annual kWh - The annual energy consumption of the other residual plug loads. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + The annual energy consumption of the other residual plug loads. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double kWh/yr false @@ -6672,7 +6672,7 @@ misc_plug_loads_other_frac_sensible Misc Plug Loads: Other Sensible Fraction - Fraction of other residual plug loads' internal gains that are sensible. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + Fraction of other residual plug loads' internal gains that are sensible. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double Frac false @@ -6681,7 +6681,7 @@ misc_plug_loads_other_frac_latent Misc Plug Loads: Other Latent Fraction - Fraction of other residual plug loads' internal gains that are latent. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + Fraction of other residual plug loads' internal gains that are latent. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double Frac false @@ -6690,7 +6690,7 @@ misc_plug_loads_other_usage_multiplier Misc Plug Loads: Other Usage Multiplier - Multiplier on the other energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + Multiplier on the other energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double false false @@ -6717,7 +6717,7 @@ misc_plug_loads_well_pump_annual_kwh Misc Plug Loads: Well Pump Annual kWh - The annual energy consumption of the well pump plug loads. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + The annual energy consumption of the well pump plug loads. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double kWh/yr false @@ -6726,7 +6726,7 @@ misc_plug_loads_well_pump_usage_multiplier Misc Plug Loads: Well Pump Usage Multiplier - Multiplier on the well pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + Multiplier on the well pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double false false @@ -6753,7 +6753,7 @@ misc_plug_loads_vehicle_annual_kwh Misc Plug Loads: Vehicle Annual kWh - The annual energy consumption of the electric vehicle plug loads. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + The annual energy consumption of the electric vehicle plug loads. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double kWh/yr false @@ -6762,7 +6762,7 @@ misc_plug_loads_vehicle_usage_multiplier Misc Plug Loads: Vehicle Usage Multiplier - Multiplier on the electric vehicle energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. + Multiplier on the electric vehicle energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-plug-loads'>HPXML Plug Loads</a>) is used. Double false false @@ -6820,7 +6820,7 @@ misc_fuel_loads_grill_annual_therm Misc Fuel Loads: Grill Annual therm - The annual energy consumption of the fuel loads grill. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. + The annual energy consumption of the fuel loads grill. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. Double therm/yr false @@ -6829,7 +6829,7 @@ misc_fuel_loads_grill_usage_multiplier Misc Fuel Loads: Grill Usage Multiplier - Multiplier on the fuel loads grill energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. + Multiplier on the fuel loads grill energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. Double false false @@ -6887,7 +6887,7 @@ misc_fuel_loads_lighting_annual_therm Misc Fuel Loads: Lighting Annual therm - The annual energy consumption of the fuel loads lighting. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>)is used. + The annual energy consumption of the fuel loads lighting. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>)is used. Double therm/yr false @@ -6896,7 +6896,7 @@ misc_fuel_loads_lighting_usage_multiplier Misc Fuel Loads: Lighting Usage Multiplier - Multiplier on the fuel loads lighting energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. + Multiplier on the fuel loads lighting energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. Double false false @@ -6954,7 +6954,7 @@ misc_fuel_loads_fireplace_annual_therm Misc Fuel Loads: Fireplace Annual therm - The annual energy consumption of the fuel loads fireplace. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. + The annual energy consumption of the fuel loads fireplace. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. Double therm/yr false @@ -6963,7 +6963,7 @@ misc_fuel_loads_fireplace_frac_sensible Misc Fuel Loads: Fireplace Sensible Fraction - Fraction of fireplace residual fuel loads' internal gains that are sensible. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. + Fraction of fireplace residual fuel loads' internal gains that are sensible. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. Double Frac false @@ -6972,7 +6972,7 @@ misc_fuel_loads_fireplace_frac_latent Misc Fuel Loads: Fireplace Latent Fraction - Fraction of fireplace residual fuel loads' internal gains that are latent. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. + Fraction of fireplace residual fuel loads' internal gains that are latent. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. Double Frac false @@ -6981,7 +6981,7 @@ misc_fuel_loads_fireplace_usage_multiplier Misc Fuel Loads: Fireplace Usage Multiplier - Multiplier on the fuel loads fireplace energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. + Multiplier on the fuel loads fireplace energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#hpxml-fuel-loads'>HPXML Fuel Loads</a>) is used. Double false false @@ -7008,7 +7008,7 @@ pool_pump_annual_kwh Pool: Pump Annual kWh - The annual energy consumption of the pool pump. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#pool-pump'>Pool Pump</a>) is used. + The annual energy consumption of the pool pump. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#pool-pump'>Pool Pump</a>) is used. Double kWh/yr false @@ -7017,7 +7017,7 @@ pool_pump_usage_multiplier Pool: Pump Usage Multiplier - Multiplier on the pool pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#pool-pump'>Pool Pump</a>) is used. + Multiplier on the pool pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#pool-pump'>Pool Pump</a>) is used. Double false false @@ -7052,7 +7052,7 @@ pool_heater_annual_kwh Pool: Heater Annual kWh - The annual energy consumption of the electric resistance pool heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#pool-heater'>Pool Heater</a>) is used. + The annual energy consumption of the electric resistance pool heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#pool-heater'>Pool Heater</a>) is used. Double kWh/yr false @@ -7061,7 +7061,7 @@ pool_heater_annual_therm Pool: Heater Annual therm - The annual energy consumption of the gas fired pool heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#pool-heater'>Pool Heater</a>) is used. + The annual energy consumption of the gas fired pool heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#pool-heater'>Pool Heater</a>) is used. Double therm/yr false @@ -7070,7 +7070,7 @@ pool_heater_usage_multiplier Pool: Heater Usage Multiplier - Multiplier on the pool heater energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#pool-heater'>Pool Heater</a>) is used. + Multiplier on the pool heater energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#pool-heater'>Pool Heater</a>) is used. Double false false @@ -7097,7 +7097,7 @@ permanent_spa_pump_annual_kwh Permanent Spa: Pump Annual kWh - The annual energy consumption of the permanent spa pump. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#permanent-spa-pump'>Permanent Spa Pump</a>) is used. + The annual energy consumption of the permanent spa pump. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#permanent-spa-pump'>Permanent Spa Pump</a>) is used. Double kWh/yr false @@ -7106,7 +7106,7 @@ permanent_spa_pump_usage_multiplier Permanent Spa: Pump Usage Multiplier - Multiplier on the permanent spa pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#permanent-spa-pump'>Permanent Spa Pump</a>) is used. + Multiplier on the permanent spa pump energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#permanent-spa-pump'>Permanent Spa Pump</a>) is used. Double false false @@ -7141,7 +7141,7 @@ permanent_spa_heater_annual_kwh Permanent Spa: Heater Annual kWh - The annual energy consumption of the electric resistance permanent spa heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#permanent-spa-heater'>Permanent Spa Heater</a>) is used. + The annual energy consumption of the electric resistance permanent spa heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#permanent-spa-heater'>Permanent Spa Heater</a>) is used. Double kWh/yr false @@ -7150,7 +7150,7 @@ permanent_spa_heater_annual_therm Permanent Spa: Heater Annual therm - The annual energy consumption of the gas fired permanent spa heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#permanent-spa-heater'>Permanent Spa Heater</a>) is used. + The annual energy consumption of the gas fired permanent spa heater. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#permanent-spa-heater'>Permanent Spa Heater</a>) is used. Double therm/yr false @@ -7159,7 +7159,7 @@ permanent_spa_heater_usage_multiplier Permanent Spa: Heater Usage Multiplier - Multiplier on the permanent spa heater energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.8.1/workflow_inputs.html#permanent-spa-heater'>Permanent Spa Heater</a>) is used. + Multiplier on the permanent spa heater energy usage that can reflect, e.g., high/low usage occupants. If not provided, the OS-HPXML default (see <a href='https://openstudio-hpxml.readthedocs.io/en/v1.10.0/workflow_inputs.html#permanent-spa-heater'>Permanent Spa Heater</a>) is used. Double false false @@ -7527,7 +7527,7 @@ README.md md readme - 25110074 + ECAEDA1E README.md.erb @@ -7544,7 +7544,7 @@ measure.rb rb script - 79912FCC + 612549F4 constants.rb @@ -7556,13 +7556,13 @@ geometry.rb rb resource - E50F00EB + 425682E4 test_build_residential_hpxml.rb rb test - 04146810 + CCBB49E7 diff --git a/BuildResidentialHPXML/resources/geometry.rb b/BuildResidentialHPXML/resources/geometry.rb index cba189f455..6bf1d4ef74 100644 --- a/BuildResidentialHPXML/resources/geometry.rb +++ b/BuildResidentialHPXML/resources/geometry.rb @@ -268,7 +268,8 @@ def self.create_single_family_detached(runner:, # make polygons polygon_floor = make_polygon(roof_nw_point, roof_ne_point, roof_se_point, roof_sw_point) side_type = nil - if roof_type == Constants::RoofTypeGable + case roof_type + when Constants::RoofTypeGable if length >= width roof_w_point = OpenStudio::Point3d.new(0, width / 2.0, z + attic_height) roof_e_point = OpenStudio::Point3d.new(length, width / 2.0, z + attic_height) @@ -285,7 +286,7 @@ def self.create_single_family_detached(runner:, polygon_e_wall = make_polygon(roof_e_point, roof_ne_point, roof_nw_point) end side_type = EPlus::SurfaceTypeWall - elsif roof_type == Constants::RoofTypeHip + when Constants::RoofTypeHip if length >= width roof_w_point = OpenStudio::Point3d.new(width / 2.0, width / 2.0, z + attic_height) roof_e_point = OpenStudio::Point3d.new(length - width / 2.0, width / 2.0, z + attic_height) @@ -330,7 +331,8 @@ def self.create_single_family_detached(runner:, surface_e_wall.setSpace(attic_space) # set these to the attic zone - if (attic_type == HPXML::AtticTypeVented) || (attic_type == HPXML::AtticTypeUnvented) + case attic_type + when HPXML::AtticTypeVented, HPXML::AtticTypeUnvented # create attic zone attic_zone = OpenStudio::Model::ThermalZone.new(model) attic_space.setThermalZone(attic_zone) @@ -340,7 +342,7 @@ def self.create_single_family_detached(runner:, attic_space_name = HPXML::LocationAtticUnvented end attic_zone.setName(attic_space_name) - elsif attic_type == HPXML::AtticTypeConditioned + when HPXML::AtticTypeConditioned attic_space.setThermalZone(conditioned_zone) attic_space_name = HPXML::LocationConditionedSpace end @@ -379,20 +381,23 @@ def self.create_single_family_detached(runner:, foundation_space = OpenStudio::Model::Space::fromFloorPrint(foundation_polygon, foundation_height, model) foundation_space = foundation_space.get assign_indexes(model: model, footprint_polygon: foundation_polygon, space: foundation_space) - if foundation_type == HPXML::FoundationTypeCrawlspaceVented + case foundation_type + when HPXML::FoundationTypeCrawlspaceVented foundation_space_name = HPXML::LocationCrawlspaceVented - elsif foundation_type == HPXML::FoundationTypeCrawlspaceUnvented + when HPXML::FoundationTypeCrawlspaceUnvented foundation_space_name = HPXML::LocationCrawlspaceUnvented - elsif foundation_type == HPXML::FoundationTypeCrawlspaceConditioned + when HPXML::FoundationTypeCrawlspaceConditioned foundation_space_name = HPXML::LocationCrawlspaceConditioned - elsif foundation_type == HPXML::FoundationTypeBasementUnconditioned + when HPXML::FoundationTypeBasementUnconditioned foundation_space_name = HPXML::LocationBasementUnconditioned - elsif foundation_type == HPXML::FoundationTypeBasementConditioned + when HPXML::FoundationTypeBasementConditioned foundation_space_name = HPXML::LocationBasementConditioned - elsif foundation_type == HPXML::FoundationTypeAmbient + when HPXML::FoundationTypeAmbient foundation_space_name = HPXML::LocationOutside - elsif foundation_type.start_with?(HPXML::FoundationTypeBellyAndWing) - foundation_space_name = HPXML::LocationManufacturedHomeUnderBelly + else + if foundation_type.start_with? HPXML::FoundationTypeBellyAndWing + foundation_space_name = HPXML::LocationManufacturedHomeUnderBelly + end end foundation_zone.setName(foundation_space_name) foundation_space.setName(foundation_space_name) @@ -814,17 +819,18 @@ def self.create_single_family_attached(model:, # create foundation zone foundation_zone = OpenStudio::Model::ThermalZone.new(model) - if foundation_type == HPXML::FoundationTypeCrawlspaceVented + case foundation_type + when HPXML::FoundationTypeCrawlspaceVented foundation_space_name = HPXML::LocationCrawlspaceVented - elsif foundation_type == HPXML::FoundationTypeCrawlspaceUnvented + when HPXML::FoundationTypeCrawlspaceUnvented foundation_space_name = HPXML::LocationCrawlspaceUnvented - elsif foundation_type == HPXML::FoundationTypeCrawlspaceConditioned + when HPXML::FoundationTypeCrawlspaceConditioned foundation_space_name = HPXML::LocationCrawlspaceConditioned - elsif foundation_type == HPXML::FoundationTypeBasementUnconditioned + when HPXML::FoundationTypeBasementUnconditioned foundation_space_name = HPXML::LocationBasementUnconditioned - elsif foundation_type == HPXML::FoundationTypeBasementConditioned + when HPXML::FoundationTypeBasementConditioned foundation_space_name = HPXML::LocationBasementConditioned - elsif foundation_type == HPXML::FoundationTypeAmbient + when HPXML::FoundationTypeAmbient foundation_space_name = HPXML::LocationOutside end foundation_zone.setName(foundation_space_name) @@ -889,7 +895,8 @@ def self.create_single_family_attached(model:, attic_space = get_attic_space(model: model, x: x, y: y, average_ceiling_height: average_ceiling_height, num_floors: num_floors, roof_pitch: roof_pitch, roof_type: roof_type, rim_joist_height: rim_joist_height) # set these to the attic zone - if (attic_type == HPXML::AtticTypeVented) || (attic_type == HPXML::AtticTypeUnvented) + case attic_type + when HPXML::AtticTypeVented, HPXML::AtticTypeUnvented # create attic zone attic_zone = OpenStudio::Model::ThermalZone.new(model) attic_space.setThermalZone(attic_zone) @@ -1106,17 +1113,18 @@ def self.create_apartment(model:, # create foundation zone foundation_zone = OpenStudio::Model::ThermalZone.new(model) - if foundation_type == HPXML::FoundationTypeCrawlspaceVented + case foundation_type + when HPXML::FoundationTypeCrawlspaceVented foundation_space_name = HPXML::LocationCrawlspaceVented - elsif foundation_type == HPXML::FoundationTypeCrawlspaceUnvented + when HPXML::FoundationTypeCrawlspaceUnvented foundation_space_name = HPXML::LocationCrawlspaceUnvented - elsif foundation_type == HPXML::FoundationTypeCrawlspaceConditioned + when HPXML::FoundationTypeCrawlspaceConditioned foundation_space_name = HPXML::LocationCrawlspaceConditioned - elsif foundation_type == HPXML::FoundationTypeBasementUnconditioned + when HPXML::FoundationTypeBasementUnconditioned foundation_space_name = HPXML::LocationBasementUnconditioned - elsif foundation_type == HPXML::FoundationTypeBasementConditioned + when HPXML::FoundationTypeBasementConditioned foundation_space_name = HPXML::LocationBasementConditioned - elsif foundation_type == HPXML::FoundationTypeAmbient + when HPXML::FoundationTypeAmbient foundation_space_name = HPXML::LocationOutside end foundation_zone.setName(foundation_space_name) @@ -1179,7 +1187,8 @@ def self.create_apartment(model:, attic_space = get_attic_space(model: model, x: x, y: y, average_ceiling_height: average_ceiling_height, num_floors: num_floors, roof_pitch: roof_pitch, roof_type: roof_type, rim_joist_height: rim_joist_height) # set these to the attic zone - if (attic_type == HPXML::AtticTypeVented) || (attic_type == HPXML::AtticTypeUnvented) + case attic_type + when HPXML::AtticTypeVented, HPXML::AtticTypeUnvented # create attic zone attic_zone = OpenStudio::Model::ThermalZone.new(model) attic_space.setThermalZone(attic_zone) @@ -1326,16 +1335,17 @@ def self.create_doors(runner:, # Convert to 3D geometry; assign to surface door_polygon = OpenStudio::Point3dVector.new - if facade == Constants::FacadeFront + case facade + when Constants::FacadeFront multx = 1 multy = 0 - elsif facade == Constants::FacadeBack + when Constants::FacadeBack multx = -1 multy = 0 - elsif facade == Constants::FacadeLeft + when Constants::FacadeLeft multx = 0 multy = -1 - elsif facade == Constants::FacadeRight + when Constants::FacadeRight multx = 0 multy = 1 end @@ -1602,19 +1612,20 @@ def self.create_windows_and_skylights(runner:, leftx = skylight_bottom_left.x lefty = skylight_bottom_left.y bottomz = skylight_bottom_left.z - if (facade == Constants::FacadeFront) || (facade == Constants::FacadeNone) + case facade + when Constants::FacadeFront, Constants::FacadeNone skylight_top_left = OpenStudio::Point3d.new(leftx, lefty + Math.cos(surface.tilt) * skylight_length, bottomz + Math.sin(surface.tilt) * skylight_length) skylight_top_right = OpenStudio::Point3d.new(leftx + skylight_width, lefty + Math.cos(surface.tilt) * skylight_length, bottomz + Math.sin(surface.tilt) * skylight_length) skylight_bottom_right = OpenStudio::Point3d.new(leftx + skylight_width, lefty, bottomz) - elsif facade == Constants::FacadeBack + when Constants::FacadeBack skylight_top_left = OpenStudio::Point3d.new(leftx, lefty - Math.cos(surface.tilt) * skylight_length, bottomz + Math.sin(surface.tilt) * skylight_length) skylight_top_right = OpenStudio::Point3d.new(leftx - skylight_width, lefty - Math.cos(surface.tilt) * skylight_length, bottomz + Math.sin(surface.tilt) * skylight_length) skylight_bottom_right = OpenStudio::Point3d.new(leftx - skylight_width, lefty, bottomz) - elsif facade == Constants::FacadeLeft + when Constants::FacadeLeft skylight_top_left = OpenStudio::Point3d.new(leftx + Math.cos(surface.tilt) * skylight_length, lefty, bottomz + Math.sin(surface.tilt) * skylight_length) skylight_top_right = OpenStudio::Point3d.new(leftx + Math.cos(surface.tilt) * skylight_length, lefty - skylight_width, bottomz + Math.sin(surface.tilt) * skylight_length) skylight_bottom_right = OpenStudio::Point3d.new(leftx, lefty - skylight_width, bottomz) - elsif facade == Constants::FacadeRight + when Constants::FacadeRight skylight_top_left = OpenStudio::Point3d.new(leftx - Math.cos(surface.tilt) * skylight_length, lefty, bottomz + Math.sin(surface.tilt) * skylight_length) skylight_top_right = OpenStudio::Point3d.new(leftx - Math.cos(surface.tilt) * skylight_length, lefty + skylight_width, bottomz + Math.sin(surface.tilt) * skylight_length) skylight_bottom_right = OpenStudio::Point3d.new(leftx, lefty + skylight_width, bottomz) @@ -1802,13 +1813,14 @@ def self.get_facade_for_surface(surface:) # @return [Double] the absolute azimuth based on relative azimuth of the facade and building orientation def self.get_azimuth_from_facade(facade:, orientation:) - if facade == Constants::FacadeFront + case facade + when Constants::FacadeFront return get_abs_azimuth(relative_azimuth: 0, building_orientation: orientation) - elsif facade == Constants::FacadeBack + when Constants::FacadeBack return get_abs_azimuth(relative_azimuth: 180, building_orientation: orientation) - elsif facade == Constants::FacadeLeft + when Constants::FacadeLeft return get_abs_azimuth(relative_azimuth: 90, building_orientation: orientation) - elsif facade == Constants::FacadeRight + when Constants::FacadeRight return get_abs_azimuth(relative_azimuth: 270, building_orientation: orientation) else fail 'Unexpected facade.' @@ -2290,16 +2302,17 @@ def self.add_window_to_wall(surface:, # Convert to 3D geometry; assign to surface window_polygon = OpenStudio::Point3dVector.new - if facade == Constants::FacadeFront + case facade + when Constants::FacadeFront multx = 1 multy = 0 - elsif facade == Constants::FacadeBack + when Constants::FacadeBack multx = -1 multy = 0 - elsif facade == Constants::FacadeLeft + when Constants::FacadeLeft multx = 0 multy = -1 - elsif facade == Constants::FacadeRight + when Constants::FacadeRight multx = 0 multy = 1 end @@ -2458,7 +2471,8 @@ def self.get_attic_space(model:, attic_height = (y_tot / 2.0) * roof_pitch + rim_joist_height # Roof always has same orientation side_type = nil - if roof_type == Constants::RoofTypeGable + case roof_type + when Constants::RoofTypeGable roof_w_point = OpenStudio::Point3d.new(0, y_peak, average_ceiling_height * num_floors + attic_height) roof_e_point = OpenStudio::Point3d.new(x, y_peak, average_ceiling_height * num_floors + attic_height) polygon_w_roof = make_polygon(roof_w_point, roof_e_point, ne_point, nw_point) @@ -2466,7 +2480,7 @@ def self.get_attic_space(model:, polygon_s_wall = make_polygon(roof_w_point, nw_point, sw_point) polygon_n_wall = make_polygon(roof_e_point, se_point, ne_point) side_type = EPlus::SurfaceTypeWall - elsif roof_type == Constants::RoofTypeHip + when Constants::RoofTypeHip if y > 0 if x <= (y + y_rear) roof_n_point = OpenStudio::Point3d.new(x / 2.0, y_rear - x / 2.0, average_ceiling_height * num_floors + attic_height) diff --git a/BuildResidentialHPXML/tests/test_build_residential_hpxml.rb b/BuildResidentialHPXML/tests/test_build_residential_hpxml.rb index 9b79b0e8b8..06a9471fff 100644 --- a/BuildResidentialHPXML/tests/test_build_residential_hpxml.rb +++ b/BuildResidentialHPXML/tests/test_build_residential_hpxml.rb @@ -386,6 +386,22 @@ def test_workflows assert_equal(31, hvac_control.seasons_cooling_end_day) end + def test_version + found_match = false + measure_xml_path = File.join(File.dirname(__FILE__), '..', 'measure.xml') + File.readlines(measure_xml_path).each do |xml_line| + next unless xml_line.include? '' + next unless xml_line.include? 'https://openstudio-hpxml.readthedocs.io' + + found_match = true + if not xml_line.include? Version::OS_HPXML_Version + puts "ERROR: Found incorrect OS-HPXML version. Manually edit the BuildResidentialHPXML/measure.rb and run 'openstudio tasks.rb update_measures' to force the measure.xml to be regenerated." + end + assert(xml_line.include? Version::OS_HPXML_Version) + end + assert(found_match) + end + private def _set_measure_argument_values(hpxml_file, args) @@ -394,7 +410,8 @@ def _set_measure_argument_values(hpxml_file, args) args['apply_validation'] = true # Base - if ['base-sfd.xml'].include? hpxml_file + case hpxml_file + when 'base-sfd.xml' args['simulation_control_timestep'] = 60 args['weather_station_epw_filepath'] = 'USA_CO_Denver.Intl.AP.725650_TMY3.epw' args['site_type'] = HPXML::SiteTypeSuburban @@ -669,10 +686,10 @@ def _set_measure_argument_values(hpxml_file, args) args['pool_heater_type'] = HPXML::HeaterTypeElectricResistance args['permanent_spa_present'] = false args['permanent_spa_heater_type'] = HPXML::HeaterTypeElectricResistance - elsif ['base-sfd2.xml'].include? hpxml_file + when 'base-sfd2.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-sfd.xml') args['whole_sfa_or_mf_building_sim'] = true - elsif ['base-sfa.xml'].include? hpxml_file + when 'base-sfa.xml' args['geometry_unit_type'] = HPXML::ResidentialTypeSFA args['geometry_unit_cfa'] = 1800.0 args['geometry_building_num_units'] = 3 @@ -686,13 +703,13 @@ def _set_measure_argument_values(hpxml_file, args) args['window_area_left'] = 0 args['window_area_right'] = 0 args['air_leakage_type'] = HPXML::InfiltrationTypeUnitTotal - elsif ['base-sfa2.xml'].include? hpxml_file + when 'base-sfa2.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-sfa.xml') args['whole_sfa_or_mf_building_sim'] = true - elsif ['base-sfa3.xml'].include? hpxml_file + when 'base-sfa3.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-sfa2.xml') args['whole_sfa_or_mf_building_sim'] = true - elsif ['base-mf.xml'].include? hpxml_file + when 'base-mf.xml' args['geometry_unit_type'] = HPXML::ResidentialTypeApartment args['geometry_unit_cfa'] = 900.0 args['geometry_foundation_type'] = HPXML::FoundationTypeBasementUnconditioned @@ -717,16 +734,16 @@ def _set_measure_argument_values(hpxml_file, args) args['ducts_number_of_return_registers'] = 1 args['door_area'] = 20.0 args['air_leakage_type'] = HPXML::InfiltrationTypeUnitTotal - elsif ['base-mf2.xml'].include? hpxml_file + when 'base-mf2.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-mf.xml') args['whole_sfa_or_mf_building_sim'] = true - elsif ['base-mf3.xml'].include? hpxml_file + when 'base-mf3.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-mf2.xml') args['whole_sfa_or_mf_building_sim'] = true - elsif ['base-mf4.xml'].include? hpxml_file + when 'base-mf4.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-mf3.xml') args['whole_sfa_or_mf_building_sim'] = true - elsif ['base-sfd-header.xml'].include? hpxml_file + when 'base-sfd-header.xml' args['software_info_program_used'] = 'Program' args['software_info_program_version'] = '1' args['schedules_unavailable_period_types'] = 'Vacancy, Power Outage' @@ -742,13 +759,14 @@ def _set_measure_argument_values(hpxml_file, args) args['emissions_fossil_fuel_units'] = 'kg/MBtu' args['emissions_natural_gas_values'] = '2' args['utility_bill_scenario_names'] = 'Bills' - elsif ['base-sfd-header-no-duplicates.xml'].include? hpxml_file + when 'base-sfd-header-no-duplicates.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-sfd-header.xml') args['whole_sfa_or_mf_building_sim'] = true end # Extras - if ['extra-auto.xml'].include? hpxml_file + case hpxml_file + when 'extra-auto.xml' args.delete('geometry_unit_num_occupants') args.delete('ducts_supply_location') args.delete('ducts_return_location') @@ -760,20 +778,20 @@ def _set_measure_argument_values(hpxml_file, args) args.delete('clothes_washer_location') args.delete('clothes_dryer_location') args.delete('refrigerator_location') - elsif ['extra-auto-duct-locations.xml'].include? hpxml_file + when 'extra-auto-duct-locations.xml' args['ducts_supply_location'] = HPXML::LocationAtticUnvented args['ducts_return_location'] = HPXML::LocationAtticUnvented - elsif ['extra-pv-roofpitch.xml'].include? hpxml_file + when 'extra-pv-roofpitch.xml' args['pv_system_module_type'] = HPXML::PVModuleTypeStandard args['pv_system_2_module_type'] = HPXML::PVModuleTypeStandard args['pv_system_array_tilt'] = 'roofpitch' args['pv_system_2_array_tilt'] = 'roofpitch+15' - elsif ['extra-dhw-solar-latitude.xml'].include? hpxml_file + when 'extra-dhw-solar-latitude.xml' args['solar_thermal_system_type'] = HPXML::SolarThermalSystemTypeHotWater args['solar_thermal_collector_tilt'] = 'Latitude-15' - elsif ['extra-second-refrigerator.xml'].include? hpxml_file + when 'extra-second-refrigerator.xml' args['extra_refrigerator_location'] = HPXML::LocationConditionedSpace - elsif ['extra-second-heating-system-portable-heater-to-heating-system.xml'].include? hpxml_file + when 'extra-second-heating-system-portable-heater-to-heating-system.xml' args['heating_system_fuel'] = HPXML::FuelTypeElectricity args['heating_system_heating_capacity'] = 48000.0 args['heating_system_fraction_heat_load_served'] = 0.75 @@ -783,7 +801,7 @@ def _set_measure_argument_values(hpxml_file, args) args['ducts_return_location'] = HPXML::LocationConditionedSpace args['heating_system_2_type'] = HPXML::HVACTypeSpaceHeater args['heating_system_2_heating_capacity'] = 16000.0 - elsif ['extra-second-heating-system-fireplace-to-heating-system.xml'].include? hpxml_file + when 'extra-second-heating-system-fireplace-to-heating-system.xml' args['heating_system_type'] = HPXML::HVACTypeElectricResistance args['heating_system_fuel'] = HPXML::FuelTypeElectricity args['heating_system_heating_efficiency'] = 1.0 @@ -792,11 +810,11 @@ def _set_measure_argument_values(hpxml_file, args) args['cooling_system_type'] = Constants::None args['heating_system_2_type'] = HPXML::HVACTypeFireplace args['heating_system_2_heating_capacity'] = 16000.0 - elsif ['extra-second-heating-system-boiler-to-heating-system.xml'].include? hpxml_file + when 'extra-second-heating-system-boiler-to-heating-system.xml' args['heating_system_type'] = HPXML::HVACTypeBoiler args['heating_system_fraction_heat_load_served'] = 0.75 args['heating_system_2_type'] = HPXML::HVACTypeBoiler - elsif ['extra-second-heating-system-portable-heater-to-heat-pump.xml'].include? hpxml_file + when 'extra-second-heating-system-portable-heater-to-heat-pump.xml' args['heating_system_type'] = Constants::None args['cooling_system_type'] = Constants::None args['heat_pump_type'] = HPXML::HVACTypeHeatPumpAirToAir @@ -810,7 +828,7 @@ def _set_measure_argument_values(hpxml_file, args) args['ducts_return_location'] = HPXML::LocationConditionedSpace args['heating_system_2_type'] = HPXML::HVACTypeSpaceHeater args['heating_system_2_heating_capacity'] = 16000.0 - elsif ['extra-second-heating-system-fireplace-to-heat-pump.xml'].include? hpxml_file + when 'extra-second-heating-system-fireplace-to-heat-pump.xml' args['heating_system_type'] = Constants::None args['cooling_system_type'] = Constants::None args['heat_pump_type'] = HPXML::HVACTypeHeatPumpMiniSplit @@ -822,7 +840,7 @@ def _set_measure_argument_values(hpxml_file, args) args['heat_pump_fraction_heat_load_served'] = 0.75 args['heating_system_2_type'] = HPXML::HVACTypeFireplace args['heating_system_2_heating_capacity'] = 16000.0 - elsif ['extra-second-heating-system-boiler-to-heat-pump.xml'].include? hpxml_file + when 'extra-second-heating-system-boiler-to-heat-pump.xml' args['heating_system_type'] = Constants::None args['cooling_system_type'] = Constants::None args['heat_pump_type'] = HPXML::HVACTypeHeatPumpGroundToAir @@ -834,15 +852,15 @@ def _set_measure_argument_values(hpxml_file, args) args['heat_pump_backup_fuel'] = HPXML::FuelTypeElectricity args['heat_pump_fraction_heat_load_served'] = 0.75 args['heating_system_2_type'] = HPXML::HVACTypeBoiler - elsif ['extra-enclosure-windows-shading.xml'].include? hpxml_file + when 'extra-enclosure-windows-shading.xml' args['window_interior_shading_winter'] = 0.99 args['window_interior_shading_summer'] = 0.01 args['window_exterior_shading_winter'] = 0.9 args['window_exterior_shading_summer'] = 0.1 - elsif ['extra-enclosure-garage-partially-protruded.xml'].include? hpxml_file + when 'extra-enclosure-garage-partially-protruded.xml' args['geometry_garage_width'] = 12 args['geometry_garage_protrusion'] = 0.5 - elsif ['extra-enclosure-garage-atticroof-conditioned.xml'].include? hpxml_file + when 'extra-enclosure-garage-atticroof-conditioned.xml' args['geometry_garage_width'] = 30.0 args['geometry_garage_protrusion'] = 1.0 args['window_area_front'] = 12.0 @@ -853,7 +871,7 @@ def _set_measure_argument_values(hpxml_file, args) args['floor_over_garage_assembly_r'] = 39.3 args['ducts_supply_location'] = HPXML::LocationGarage args['ducts_return_location'] = HPXML::LocationGarage - elsif ['extra-enclosure-atticroof-conditioned-eaves-gable.xml'].include? hpxml_file + when 'extra-enclosure-atticroof-conditioned-eaves-gable.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeSlab args['geometry_foundation_height'] = 0.0 args['geometry_foundation_height_above_grade'] = 0.0 @@ -864,26 +882,26 @@ def _set_measure_argument_values(hpxml_file, args) args['geometry_eaves_depth'] = 2 args['ducts_supply_location'] = HPXML::LocationUnderSlab args['ducts_return_location'] = HPXML::LocationUnderSlab - elsif ['extra-enclosure-atticroof-conditioned-eaves-hip.xml'].include? hpxml_file + when 'extra-enclosure-atticroof-conditioned-eaves-hip.xml' args['geometry_roof_type'] = Constants::RoofTypeHip - elsif ['extra-gas-pool-heater-with-zero-kwh.xml'].include? hpxml_file + when 'extra-gas-pool-heater-with-zero-kwh.xml' args['pool_present'] = true args['pool_heater_type'] = HPXML::HeaterTypeGas args['pool_heater_annual_kwh'] = 0 - elsif ['extra-gas-hot-tub-heater-with-zero-kwh.xml'].include? hpxml_file + when 'extra-gas-hot-tub-heater-with-zero-kwh.xml' args['permanent_spa_present'] = true args['permanent_spa_heater_type'] = HPXML::HeaterTypeGas args['permanent_spa_heater_annual_kwh'] = 0 - elsif ['extra-no-rim-joists.xml'].include? hpxml_file + when 'extra-no-rim-joists.xml' args.delete('geometry_rim_joist_height') args.delete('rim_joist_assembly_r') - elsif ['extra-iecc-zone-different-than-epw.xml'].include? hpxml_file + when 'extra-iecc-zone-different-than-epw.xml' args['site_iecc_zone'] = '6B' - elsif ['extra-state-code-different-than-epw.xml'].include? hpxml_file + when 'extra-state-code-different-than-epw.xml' args['site_state_code'] = 'WY' - elsif ['extra-time-zone-different-than-epw.xml'].include? hpxml_file + when 'extra-time-zone-different-than-epw.xml' args['site_time_zone_utc_offset'] = '-6' - elsif ['extra-emissions-fossil-fuel-factors.xml'].include? hpxml_file + when 'extra-emissions-fossil-fuel-factors.xml' args['emissions_scenario_names'] = 'Scenario1, Scenario2' args['emissions_types'] = 'CO2e, SO2' args['emissions_electricity_units'] = "#{HPXML::EmissionsScenario::UnitsKgPerMWh}, #{HPXML::EmissionsScenario::UnitsLbPerMWh}" @@ -894,7 +912,7 @@ def _set_measure_argument_values(hpxml_file, args) args['emissions_fuel_oil_values'] = '161.0, 0.0015' args['emissions_coal_values'] = '211.1, 0.0020' args['emissions_wood_values'] = '200.0, 0.0025' - elsif ['extra-bills-fossil-fuel-rates.xml'].include? hpxml_file + when 'extra-bills-fossil-fuel-rates.xml' args['utility_bill_scenario_names'] = 'Scenario1, Scenario2' args['utility_bill_propane_fixed_charges'] = '1, 2' args['utility_bill_propane_marginal_rates'] = '3, 4' @@ -906,38 +924,38 @@ def _set_measure_argument_values(hpxml_file, args) args['utility_bill_wood_marginal_rates'] = '14, 15' args['utility_bill_wood_pellets_fixed_charges'] = '16, 17' args['utility_bill_wood_pellets_marginal_rates'] = '18, 19' - elsif ['extra-seasons-building-america.xml'].include? hpxml_file + when 'extra-seasons-building-america.xml' args['hvac_control_heating_season_period'] = Constants::BuildingAmerica args['hvac_control_cooling_season_period'] = Constants::BuildingAmerica - elsif ['extra-ducts-crawlspace.xml'].include? hpxml_file + when 'extra-ducts-crawlspace.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceUnvented args['geometry_foundation_height'] = 4 args['floor_over_foundation_assembly_r'] = 18.7 args['foundation_wall_insulation_distance_to_bottom'] = 4 args['ducts_supply_location'] = HPXML::LocationCrawlspace args['ducts_return_location'] = HPXML::LocationCrawlspace - elsif ['extra-ducts-attic.xml'].include? hpxml_file + when 'extra-ducts-attic.xml' args['ducts_supply_location'] = HPXML::LocationAttic args['ducts_return_location'] = HPXML::LocationAttic - elsif ['extra-water-heater-crawlspace.xml'].include? hpxml_file + when 'extra-water-heater-crawlspace.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceUnvented args['geometry_foundation_height'] = 4 args['floor_over_foundation_assembly_r'] = 18.7 args['foundation_wall_insulation_distance_to_bottom'] = 4 args['water_heater_location'] = HPXML::LocationCrawlspace - elsif ['extra-water-heater-attic.xml'].include? hpxml_file + when 'extra-water-heater-attic.xml' args['water_heater_location'] = HPXML::LocationAttic - elsif ['extra-battery-crawlspace.xml'].include? hpxml_file + when 'extra-battery-crawlspace.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceUnvented args['geometry_foundation_height'] = 4 args['floor_over_foundation_assembly_r'] = 18.7 args['foundation_wall_insulation_distance_to_bottom'] = 4 args['battery_present'] = true args['battery_location'] = HPXML::LocationCrawlspace - elsif ['extra-battery-attic.xml'].include? hpxml_file + when 'extra-battery-attic.xml' args['battery_present'] = true args['battery_location'] = HPXML::LocationAttic - elsif ['extra-detailed-performance-autosize.xml'].include? hpxml_file + when 'extra-detailed-performance-autosize.xml' args['heating_system_type'] = Constants::None args['cooling_system_type'] = Constants::None args['heat_pump_type'] = HPXML::HVACTypeHeatPumpAirToAir @@ -958,214 +976,202 @@ def _set_measure_argument_values(hpxml_file, args) args['hvac_perf_data_cooling_max_speed_capacities'] = '1.0, 1.11' args['hvac_perf_data_cooling_min_speed_cops'] = '4.47, 6.34' args['hvac_perf_data_cooling_max_speed_cops'] = '2.71, 3.53' - elsif ['extra-power-outage-periods.xml'].include? hpxml_file + when 'extra-power-outage-periods.xml' args['schedules_unavailable_period_types'] = 'Power Outage, Power Outage' args['schedules_unavailable_period_dates'] = 'Jan 1 - Jan 5, Jan 7 - Jan 9' - elsif ['extra-sfa-atticroof-flat.xml'].include? hpxml_file + when 'extra-sfa-atticroof-flat.xml' args['geometry_attic_type'] = HPXML::AtticTypeFlatRoof args['ducts_supply_leakage_to_outside_value'] = 0.0 args['ducts_return_leakage_to_outside_value'] = 0.0 args['ducts_supply_location'] = HPXML::LocationBasementConditioned args['ducts_return_location'] = HPXML::LocationBasementConditioned - elsif ['extra-sfa-atticroof-conditioned-eaves-gable.xml'].include? hpxml_file + when 'extra-sfa-atticroof-conditioned-eaves-gable.xml' args['geometry_unit_num_floors_above_grade'] = 2 args['geometry_attic_type'] = HPXML::AtticTypeConditioned args['geometry_eaves_depth'] = 2 args['ducts_supply_location'] = HPXML::LocationConditionedSpace args['ducts_return_location'] = HPXML::LocationConditionedSpace - elsif ['extra-sfa-atticroof-conditioned-eaves-hip.xml'].include? hpxml_file + when 'extra-sfa-atticroof-conditioned-eaves-hip.xml' args['geometry_roof_type'] = Constants::RoofTypeHip - elsif ['extra-mf-eaves.xml'].include? hpxml_file + when 'extra-mf-eaves.xml' args['geometry_eaves_depth'] = 2 - elsif ['extra-sfa-slab.xml'].include? hpxml_file + when 'extra-sfa-slab.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeSlab args['geometry_foundation_height'] = 0.0 args['geometry_foundation_height_above_grade'] = 0.0 args.delete('foundation_wall_insulation_distance_to_bottom') - elsif ['extra-sfa-vented-crawlspace.xml'].include? hpxml_file + when 'extra-sfa-vented-crawlspace.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceVented args['geometry_foundation_height'] = 4.0 args['floor_over_foundation_assembly_r'] = 18.7 args['foundation_wall_insulation_distance_to_bottom'] = 4.0 - elsif ['extra-sfa-unvented-crawlspace.xml'].include? hpxml_file + when 'extra-sfa-unvented-crawlspace.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceUnvented args['geometry_foundation_height'] = 4.0 args['floor_over_foundation_assembly_r'] = 18.7 args['foundation_wall_insulation_distance_to_bottom'] = 4.0 - elsif ['extra-sfa-conditioned-crawlspace.xml'].include? hpxml_file + when 'extra-sfa-conditioned-crawlspace.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceConditioned args['geometry_foundation_height'] = 4.0 args['floor_over_foundation_assembly_r'] = 2.1 args['foundation_wall_insulation_distance_to_bottom'] = 4.0 - elsif ['extra-sfa-unconditioned-basement.xml'].include? hpxml_file + when 'extra-sfa-unconditioned-basement.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeBasementUnconditioned args['floor_over_foundation_assembly_r'] = 18.7 args['foundation_wall_insulation_r'] = 0 args['foundation_wall_insulation_distance_to_bottom'] = 0.0 - elsif ['extra-sfa-ambient.xml'].include? hpxml_file + when 'extra-sfa-ambient.xml' args['geometry_unit_cfa'] = 900.0 args['geometry_foundation_type'] = HPXML::FoundationTypeAmbient args.delete('geometry_rim_joist_height') args['floor_over_foundation_assembly_r'] = 18.7 args.delete('rim_joist_assembly_r') args['misc_plug_loads_other_annual_kwh'] = 1228.5 - elsif ['extra-sfa-rear-units.xml'].include? hpxml_file + when 'extra-sfa-rear-units.xml' args['geometry_building_num_units'] = 4 - elsif ['extra-sfa-exterior-corridor.xml'].include? hpxml_file + when 'extra-sfa-exterior-corridor.xml' args['geometry_building_num_units'] = 4 - elsif ['extra-sfa-slab-middle.xml', - 'extra-sfa-vented-crawlspace-middle.xml', - 'extra-sfa-unvented-crawlspace-middle.xml', - 'extra-sfa-unconditioned-basement-middle.xml'].include? hpxml_file + when 'extra-sfa-slab-middle.xml', 'extra-sfa-vented-crawlspace-middle.xml', + 'extra-sfa-unvented-crawlspace-middle.xml', 'extra-sfa-unconditioned-basement-middle.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_unit_right_wall_is_adiabatic'] = true - elsif ['extra-sfa-slab-right.xml', - 'extra-sfa-vented-crawlspace-right.xml', - 'extra-sfa-unvented-crawlspace-right.xml', - 'extra-sfa-unconditioned-basement-right.xml'].include? hpxml_file + when 'extra-sfa-slab-right.xml', 'extra-sfa-vented-crawlspace-right.xml', + 'extra-sfa-unvented-crawlspace-right.xml', 'extra-sfa-unconditioned-basement-right.xml' args['geometry_unit_left_wall_is_adiabatic'] = true - elsif ['extra-mf-atticroof-flat.xml'].include? hpxml_file + when 'extra-mf-atticroof-flat.xml' args['geometry_attic_type'] = HPXML::AtticTypeFlatRoof - elsif ['extra-mf-atticroof-vented.xml'].include? hpxml_file + when 'extra-mf-atticroof-vented.xml' args['geometry_attic_type'] = HPXML::AtticTypeVented - elsif ['extra-mf-slab.xml'].include? hpxml_file + when 'extra-mf-slab.xml' args['geometry_building_num_units'] = 18 args['geometry_foundation_type'] = HPXML::FoundationTypeSlab args['geometry_foundation_height'] = 0.0 args['geometry_foundation_height_above_grade'] = 0.0 args.delete('foundation_wall_insulation_distance_to_bottom') - elsif ['extra-mf-vented-crawlspace.xml'].include? hpxml_file + when 'extra-mf-vented-crawlspace.xml' args['geometry_building_num_units'] = 18 args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceVented args['geometry_foundation_height'] = 4.0 args['floor_over_foundation_assembly_r'] = 18.7 args['foundation_wall_insulation_distance_to_bottom'] = 4.0 - elsif ['extra-mf-unvented-crawlspace.xml'].include? hpxml_file + when 'extra-mf-unvented-crawlspace.xml' args['geometry_building_num_units'] = 18 args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceUnvented args['geometry_foundation_height'] = 4.0 args['floor_over_foundation_assembly_r'] = 18.7 args['foundation_wall_insulation_distance_to_bottom'] = 4.0 - elsif ['extra-mf-ambient.xml'].include? hpxml_file + when 'extra-mf-ambient.xml' args['geometry_unit_cfa'] = 450.0 args['geometry_foundation_type'] = HPXML::FoundationTypeAmbient args.delete('geometry_rim_joist_height') args['floor_over_foundation_assembly_r'] = 18.7 args.delete('rim_joist_assembly_r') args['misc_plug_loads_other_annual_kwh'] = 1228.5 - elsif ['extra-mf-rear-units.xml'].include? hpxml_file + when 'extra-mf-rear-units.xml' args['geometry_building_num_units'] = 18 - elsif ['extra-mf-exterior-corridor.xml'].include? hpxml_file + when 'extra-mf-exterior-corridor.xml' args['geometry_building_num_units'] = 18 - elsif ['extra-mf-slab-left-bottom.xml', - 'extra-mf-vented-crawlspace-left-bottom.xml', - 'extra-mf-unvented-crawlspace-left-bottom.xml'].include? hpxml_file + when 'extra-mf-slab-left-bottom.xml', 'extra-mf-vented-crawlspace-left-bottom.xml', + 'extra-mf-unvented-crawlspace-left-bottom.xml' args['geometry_unit_right_wall_is_adiabatic'] = true args['geometry_attic_type'] = HPXML::AtticTypeBelowApartment - elsif ['extra-mf-slab-left-middle.xml', - 'extra-mf-vented-crawlspace-left-middle.xml', - 'extra-mf-unvented-crawlspace-left-middle.xml'].include? hpxml_file + when 'extra-mf-slab-left-middle.xml', 'extra-mf-vented-crawlspace-left-middle.xml', + 'extra-mf-unvented-crawlspace-left-middle.xml' args['geometry_unit_right_wall_is_adiabatic'] = true args['geometry_attic_type'] = HPXML::AtticTypeBelowApartment args['geometry_foundation_type'] = HPXML::FoundationTypeAboveApartment - elsif ['extra-mf-slab-left-top.xml', - 'extra-mf-vented-crawlspace-left-top.xml', - 'extra-mf-unvented-crawlspace-left-top.xml'].include? hpxml_file + when 'extra-mf-slab-left-top.xml', 'extra-mf-vented-crawlspace-left-top.xml', + 'extra-mf-unvented-crawlspace-left-top.xml' args['geometry_unit_right_wall_is_adiabatic'] = true args['geometry_foundation_type'] = HPXML::FoundationTypeAboveApartment - elsif ['extra-mf-slab-middle-bottom.xml', - 'extra-mf-vented-crawlspace-middle-bottom.xml', - 'extra-mf-unvented-crawlspace-middle-bottom.xml'].include? hpxml_file + when 'extra-mf-slab-middle-bottom.xml', 'extra-mf-vented-crawlspace-middle-bottom.xml', + 'extra-mf-unvented-crawlspace-middle-bottom.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_unit_right_wall_is_adiabatic'] = true args['geometry_attic_type'] = HPXML::AtticTypeBelowApartment - elsif ['extra-mf-slab-middle-middle.xml', - 'extra-mf-vented-crawlspace-middle-middle.xml', - 'extra-mf-unvented-crawlspace-middle-middle.xml'].include? hpxml_file + when 'extra-mf-slab-middle-middle.xml', 'extra-mf-vented-crawlspace-middle-middle.xml', + 'extra-mf-unvented-crawlspace-middle-middle.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_unit_right_wall_is_adiabatic'] = true args['geometry_attic_type'] = HPXML::AtticTypeBelowApartment args['geometry_foundation_type'] = HPXML::FoundationTypeAboveApartment - elsif ['extra-mf-slab-middle-top.xml', - 'extra-mf-vented-crawlspace-middle-top.xml', - 'extra-mf-unvented-crawlspace-middle-top.xml'].include? hpxml_file + when 'extra-mf-slab-middle-top.xml', 'extra-mf-vented-crawlspace-middle-top.xml', + 'extra-mf-unvented-crawlspace-middle-top.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_unit_right_wall_is_adiabatic'] = true args['geometry_foundation_type'] = HPXML::FoundationTypeAboveApartment - elsif ['extra-mf-slab-right-bottom.xml', - 'extra-mf-vented-crawlspace-right-bottom.xml', - 'extra-mf-unvented-crawlspace-right-bottom.xml'].include? hpxml_file + when 'extra-mf-slab-right-bottom.xml', 'extra-mf-vented-crawlspace-right-bottom.xml', + 'extra-mf-unvented-crawlspace-right-bottom.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_attic_type'] = HPXML::AtticTypeBelowApartment - elsif ['extra-mf-slab-right-middle.xml', - 'extra-mf-vented-crawlspace-right-middle.xml', - 'extra-mf-unvented-crawlspace-right-middle.xml'].include? hpxml_file + when 'extra-mf-slab-right-middle.xml', 'extra-mf-vented-crawlspace-right-middle.xml', + 'extra-mf-unvented-crawlspace-right-middle.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_attic_type'] = HPXML::AtticTypeBelowApartment args['geometry_foundation_type'] = HPXML::FoundationTypeAboveApartment - elsif ['extra-mf-slab-right-top.xml', - 'extra-mf-vented-crawlspace-right-top.xml', - 'extra-mf-unvented-crawlspace-right-top.xml'].include? hpxml_file + when 'extra-mf-slab-right-top.xml', 'extra-mf-vented-crawlspace-right-top.xml', + 'extra-mf-unvented-crawlspace-right-top.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_foundation_type'] = HPXML::FoundationTypeAboveApartment - elsif ['extra-mf-slab-rear-units.xml', - 'extra-mf-vented-crawlspace-rear-units.xml', - 'extra-mf-unvented-crawlspace-rear-units.xml', - 'extra-mf-slab-left-bottom-rear-units.xml', - 'extra-mf-slab-left-middle-rear-units.xml', - 'extra-mf-slab-left-top-rear-units.xml', - 'extra-mf-slab-middle-bottom-rear-units.xml', - 'extra-mf-slab-middle-middle-rear-units.xml', - 'extra-mf-slab-middle-top-rear-units.xml', - 'extra-mf-slab-right-bottom-rear-units.xml', - 'extra-mf-slab-right-middle-rear-units.xml', - 'extra-mf-slab-right-top-rear-units.xml', - 'extra-mf-vented-crawlspace-left-bottom-rear-units.xml', - 'extra-mf-vented-crawlspace-left-middle-rear-units.xml', - 'extra-mf-vented-crawlspace-left-top-rear-units.xml', - 'extra-mf-vented-crawlspace-middle-bottom-rear-units.xml', - 'extra-mf-vented-crawlspace-middle-middle-rear-units.xml', - 'extra-mf-vented-crawlspace-middle-top-rear-units.xml', - 'extra-mf-vented-crawlspace-right-bottom-rear-units.xml', - 'extra-mf-vented-crawlspace-right-middle-rear-units.xml', - 'extra-mf-vented-crawlspace-right-top-rear-units.xml', + when 'extra-mf-slab-rear-units.xml', + 'extra-mf-vented-crawlspace-rear-units.xml', + 'extra-mf-unvented-crawlspace-rear-units.xml', + 'extra-mf-slab-left-bottom-rear-units.xml', + 'extra-mf-slab-left-middle-rear-units.xml', + 'extra-mf-slab-left-top-rear-units.xml', + 'extra-mf-slab-middle-bottom-rear-units.xml', + 'extra-mf-slab-middle-middle-rear-units.xml', + 'extra-mf-slab-middle-top-rear-units.xml', + 'extra-mf-slab-right-bottom-rear-units.xml', + 'extra-mf-slab-right-middle-rear-units.xml', + 'extra-mf-slab-right-top-rear-units.xml', + 'extra-mf-vented-crawlspace-left-bottom-rear-units.xml', + 'extra-mf-vented-crawlspace-left-middle-rear-units.xml', + 'extra-mf-vented-crawlspace-left-top-rear-units.xml', + 'extra-mf-vented-crawlspace-middle-bottom-rear-units.xml', + 'extra-mf-vented-crawlspace-middle-middle-rear-units.xml', + 'extra-mf-vented-crawlspace-middle-top-rear-units.xml', + 'extra-mf-vented-crawlspace-right-bottom-rear-units.xml', + 'extra-mf-vented-crawlspace-right-middle-rear-units.xml', + 'extra-mf-vented-crawlspace-right-top-rear-units.xml', 'extra-mf-unvented-crawlspace-left-bottom-rear-units.xml', - 'extra-mf-unvented-crawlspace-left-middle-rear-units.xml', + 'extra-mf-unvented-crawlspace-left-middle-rear-units.xml', 'extra-mf-unvented-crawlspace-left-top-rear-units.xml', 'extra-mf-unvented-crawlspace-middle-bottom-rear-units.xml', 'extra-mf-unvented-crawlspace-middle-middle-rear-units.xml', 'extra-mf-unvented-crawlspace-middle-top-rear-units.xml', 'extra-mf-unvented-crawlspace-right-bottom-rear-units.xml', 'extra-mf-unvented-crawlspace-right-middle-rear-units.xml', - 'extra-mf-unvented-crawlspace-right-top-rear-units.xml'].include? hpxml_file + 'extra-mf-unvented-crawlspace-right-top-rear-units.xml' args['geometry_unit_front_wall_is_adiabatic'] = true end # Error - if ['error-heating-system-and-heat-pump.xml'].include? hpxml_file + case hpxml_file + when 'error-heating-system-and-heat-pump.xml' args['cooling_system_type'] = Constants::None args['heat_pump_type'] = HPXML::HVACTypeHeatPumpAirToAir - elsif ['error-cooling-system-and-heat-pump.xml'].include? hpxml_file + when 'error-cooling-system-and-heat-pump.xml' args['heating_system_type'] = Constants::None args['heat_pump_type'] = HPXML::HVACTypeHeatPumpAirToAir - elsif ['error-sfd-conditioned-basement-zero-foundation-height.xml'].include? hpxml_file + when 'error-sfd-conditioned-basement-zero-foundation-height.xml' args['geometry_foundation_height'] = 0.0 args.delete('foundation_wall_insulation_distance_to_bottom') - elsif ['error-sfd-adiabatic-walls.xml'].include? hpxml_file + when 'error-sfd-adiabatic-walls.xml' args['geometry_unit_left_wall_is_adiabatic'] = true - elsif ['error-mf-conditioned-basement'].include? hpxml_file + when 'error-mf-conditioned-basement' args['geometry_foundation_type'] = HPXML::FoundationTypeBasementConditioned - elsif ['error-mf-conditioned-crawlspace'].include? hpxml_file + when 'error-mf-conditioned-crawlspace' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceConditioned - elsif ['error-mf-bottom-crawlspace-zero-foundation-height.xml'].include? hpxml_file + when 'error-mf-bottom-crawlspace-zero-foundation-height.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceUnvented args['geometry_foundation_height'] = 0.0 args['geometry_attic_type'] = HPXML::AtticTypeBelowApartment args.delete('foundation_wall_insulation_distance_to_bottom') - elsif ['error-second-heating-system-but-no-primary-heating.xml'].include? hpxml_file + when 'error-second-heating-system-but-no-primary-heating.xml' args['heating_system_type'] = Constants::None args['heating_system_2_type'] = HPXML::HVACTypeFireplace - elsif ['error-second-heating-system-ducted-with-ducted-primary-heating.xml'].include? hpxml_file + when 'error-second-heating-system-ducted-with-ducted-primary-heating.xml' args['heating_system_type'] = Constants::None args['cooling_system_type'] = Constants::None args['heat_pump_type'] = HPXML::HVACTypeHeatPumpMiniSplit @@ -1173,119 +1179,119 @@ def _set_measure_argument_values(hpxml_file, args) args['heat_pump_is_ducted'] = true args['heat_pump_backup_type'] = HPXML::HeatPumpBackupTypeSeparate args['heating_system_2_type'] = HPXML::HVACTypeFurnace - elsif ['error-sfa-no-building-num-units.xml'].include? hpxml_file + when 'error-sfa-no-building-num-units.xml' args.delete('geometry_building_num_units') - elsif ['error-sfa-above-apartment.xml'].include? hpxml_file + when 'error-sfa-above-apartment.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeAboveApartment - elsif ['error-sfa-below-apartment.xml'].include? hpxml_file + when 'error-sfa-below-apartment.xml' args['geometry_attic_type'] = HPXML::AtticTypeBelowApartment - elsif ['error-sfa-all-adiabatic-walls.xml'].include? hpxml_file + when 'error-sfa-all-adiabatic-walls.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_unit_right_wall_is_adiabatic'] = true args['geometry_unit_front_wall_is_adiabatic'] = true args['geometry_unit_back_wall_is_adiabatic'] = true - elsif ['error-mf-no-building-num-units.xml'].include? hpxml_file + when 'error-mf-no-building-num-units.xml' args.delete('geometry_building_num_units') - elsif ['error-mf-all-adiabatic-walls.xml'].include? hpxml_file + when 'error-mf-all-adiabatic-walls.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_unit_right_wall_is_adiabatic'] = true args['geometry_unit_front_wall_is_adiabatic'] = true args['geometry_unit_back_wall_is_adiabatic'] = true - elsif ['error-mf-two-stories.xml'].include? hpxml_file + when 'error-mf-two-stories.xml' args['geometry_unit_num_floors_above_grade'] = 2 - elsif ['error-mf-conditioned-attic.xml'].include? hpxml_file + when 'error-mf-conditioned-attic.xml' args['geometry_attic_type'] = HPXML::AtticTypeConditioned - elsif ['error-dhw-indirect-without-boiler.xml'].include? hpxml_file + when 'error-dhw-indirect-without-boiler.xml' args['water_heater_type'] = HPXML::WaterHeaterTypeCombiStorage - elsif ['error-conditioned-attic-with-one-floor-above-grade.xml'].include? hpxml_file + when 'error-conditioned-attic-with-one-floor-above-grade.xml' args['geometry_attic_type'] = HPXML::AtticTypeConditioned args['ceiling_assembly_r'] = 0.0 - elsif ['error-sfd-with-shared-system.xml'].include? hpxml_file + when 'error-sfd-with-shared-system.xml' args['heating_system_type'] = "Shared #{HPXML::HVACTypeBoiler} w/ Baseboard" - elsif ['error-rim-joist-height-but-no-assembly-r.xml'].include? hpxml_file + when 'error-rim-joist-height-but-no-assembly-r.xml' args.delete('rim_joist_assembly_r') - elsif ['error-rim-joist-assembly-r-but-no-height.xml'].include? hpxml_file + when 'error-rim-joist-assembly-r-but-no-height.xml' args.delete('geometry_rim_joist_height') - elsif ['error-unavailable-period-args-not-all-specified'].include? hpxml_file + when 'error-unavailable-period-args-not-all-specified' args['schedules_unavailable_period_types'] = 'Vacancy' - elsif ['error-unavailable-period-args-not-all-same-size.xml'].include? hpxml_file + when 'error-unavailable-period-args-not-all-same-size.xml' args['schedules_unavailable_period_types'] = 'Vacancy, Power Outage' args['schedules_unavailable_period_dates'] = 'Jan 1 - Jan 5, Jan 7 - Jan 9' args['schedules_unavailable_period_window_natvent_availabilities'] = HPXML::ScheduleRegular - elsif ['error-unavailable-period-window-natvent-invalid.xml'].include? hpxml_file + when 'error-unavailable-period-window-natvent-invalid.xml' args['schedules_unavailable_period_types'] = 'Power Outage' args['schedules_unavailable_period_dates'] = 'Jan 7 - Jan 9' args['schedules_unavailable_period_window_natvent_availabilities'] = 'invalid' - elsif ['error-heating-perf-data-not-all-specified.xml'].include? hpxml_file + when 'error-heating-perf-data-not-all-specified.xml' args['hvac_perf_data_heating_outdoor_temperatures'] = '47.0' - elsif ['error-heating-perf-data-not-all-same-size.xml'].include? hpxml_file + when 'error-heating-perf-data-not-all-same-size.xml' args['hvac_perf_data_heating_outdoor_temperatures'] = '47.0' args['hvac_perf_data_heating_min_speed_capacities'] = '10000, 4200' args['hvac_perf_data_heating_max_speed_capacities'] = '36000, 24800' args['hvac_perf_data_heating_min_speed_cops'] = '4.73, 1.84' args['hvac_perf_data_heating_max_speed_cops'] = '3.44, 2.66' - elsif ['error-cooling-perf-data-not-all-specified.xml'].include? hpxml_file + when 'error-cooling-perf-data-not-all-specified.xml' args['hvac_perf_data_cooling_outdoor_temperatures'] = '95.0' - elsif ['error-cooling-perf-data-not-all-same-size.xml'].include? hpxml_file + when 'error-cooling-perf-data-not-all-same-size.xml' args['hvac_perf_data_cooling_outdoor_temperatures'] = '95.0' args['hvac_perf_data_cooling_min_speed_capacities'] = '11700, 13200' args['hvac_perf_data_cooling_max_speed_capacities'] = '36000, 40000' args['hvac_perf_data_cooling_min_speed_cops'] = '4.47, 6.34' args['hvac_perf_data_cooling_max_speed_cops'] = '2.71, 3.53' - elsif ['error-emissions-args-not-all-specified.xml'].include? hpxml_file + when 'error-emissions-args-not-all-specified.xml' args['emissions_scenario_names'] = 'Scenario1' - elsif ['error-emissions-args-not-all-same-size.xml'].include? hpxml_file + when 'error-emissions-args-not-all-same-size.xml' args['emissions_scenario_names'] = 'Scenario1' args['emissions_types'] = 'CO2e,CO2e' args['emissions_electricity_units'] = HPXML::EmissionsScenario::UnitsLbPerMWh args['emissions_electricity_values_or_filepaths'] = '../../HPXMLtoOpenStudio/resources/data/cambium/LRMER_MidCase.csv' - elsif ['error-emissions-natural-gas-args-not-all-specified.xml'].include? hpxml_file + when 'error-emissions-natural-gas-args-not-all-specified.xml' args['emissions_natural_gas_values'] = '117.6' - elsif ['error-bills-args-not-all-same-size.xml'].include? hpxml_file + when 'error-bills-args-not-all-same-size.xml' args['utility_bill_scenario_names'] = 'Scenario1' args['utility_bill_electricity_fixed_charges'] = '1' args['utility_bill_electricity_marginal_rates'] = '2,2' - elsif ['error-invalid-aspect-ratio.xml'].include? hpxml_file + when 'error-invalid-aspect-ratio.xml' args['geometry_unit_aspect_ratio'] = -1 - elsif ['error-negative-foundation-height.xml'].include? hpxml_file + when 'error-negative-foundation-height.xml' args['geometry_foundation_height'] = -8 - elsif ['error-too-many-floors.xml'].include? hpxml_file + when 'error-too-many-floors.xml' args['geometry_unit_num_floors_above_grade'] = 7 - elsif ['error-invalid-garage-protrusion.xml'].include? hpxml_file + when 'error-invalid-garage-protrusion.xml' args['geometry_garage_protrusion'] = 1.5 - elsif ['error-sfa-no-non-adiabatic-walls.xml'].include? hpxml_file + when 'error-sfa-no-non-adiabatic-walls.xml' args['geometry_unit_left_wall_is_adiabatic'] = true args['geometry_unit_front_wall_is_adiabatic'] = true args['geometry_unit_back_wall_is_adiabatic'] = true - elsif ['error-hip-roof-and-protruding-garage.xml'].include? hpxml_file + when 'error-hip-roof-and-protruding-garage.xml' args['geometry_roof_type'] = Constants::RoofTypeHip args['geometry_garage_width'] = 12 args['geometry_garage_protrusion'] = 0.5 - elsif ['error-protruding-garage-under-gable-roof.xml'].include? hpxml_file + when 'error-protruding-garage-under-gable-roof.xml' args['geometry_unit_aspect_ratio'] = 0.5 args['geometry_garage_width'] = 12 args['geometry_garage_protrusion'] = 0.5 - elsif ['error-ambient-with-garage.xml'].include? hpxml_file + when 'error-ambient-with-garage.xml' args['geometry_garage_width'] = 12 args['geometry_foundation_type'] = HPXML::FoundationTypeAmbient - elsif ['error-invalid-door-area.xml'].include? hpxml_file + when 'error-invalid-door-area.xml' args['door_area'] = -10 - elsif ['error-invalid-window-aspect-ratio.xml'].include? hpxml_file + when 'error-invalid-window-aspect-ratio.xml' args['window_aspect_ratio'] = 0 - elsif ['error-garage-too-wide.xml'].include? hpxml_file + when 'error-garage-too-wide.xml' args['geometry_garage_width'] = 72 - elsif ['error-garage-too-deep.xml'].include? hpxml_file + when 'error-garage-too-deep.xml' args['geometry_garage_width'] = 12 args['geometry_garage_depth'] = 40 - elsif ['error-vented-attic-with-zero-floor-insulation.xml'].include? hpxml_file + when 'error-vented-attic-with-zero-floor-insulation.xml' args['ceiling_assembly_r'] = 0 - elsif ['error-different-software-program.xml'].include? hpxml_file + when 'error-different-software-program.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-sfd-header.xml') args['software_info_program_used'] = 'Program2' args['software_info_program_version'] = '2' args['emissions_scenario_names'] = 'Emissions2' args['utility_bill_scenario_names'] = 'Bills2' - elsif ['error-different-simulation-control.xml'].include? hpxml_file + when 'error-different-simulation-control.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-sfd-header.xml') args['simulation_control_timestep'] = 10 args['simulation_control_run_period'] = 'Jan 2 - Dec 30' @@ -1293,63 +1299,64 @@ def _set_measure_argument_values(hpxml_file, args) args['simulation_control_temperature_capacitance_multiplier'] = 2.0 args['emissions_scenario_names'] = 'Emissions2' args['utility_bill_scenario_names'] = 'Bills2' - elsif ['error-same-emissions-scenario-name.xml'].include? hpxml_file + when 'error-same-emissions-scenario-name.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-sfd-header.xml') args['emissions_electricity_values_or_filepaths'] = '2' - elsif ['error-same-utility-bill-scenario-name.xml'].include? hpxml_file + when 'error-same-utility-bill-scenario-name.xml' args['existing_hpxml_path'] = File.join(File.dirname(__FILE__), 'extra_files/base-sfd-header.xml') args['utility_bill_electricity_fixed_charges'] = '13.0' end # Warning - if ['warning-non-electric-heat-pump-water-heater.xml'].include? hpxml_file + case hpxml_file + when 'warning-non-electric-heat-pump-water-heater.xml' args['water_heater_type'] = HPXML::WaterHeaterTypeHeatPump args['water_heater_fuel_type'] = HPXML::FuelTypeNaturalGas args['water_heater_efficiency'] = 2.3 - elsif ['warning-sfd-slab-non-zero-foundation-height.xml'].include? hpxml_file + when 'warning-sfd-slab-non-zero-foundation-height.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeSlab args['geometry_foundation_height_above_grade'] = 0.0 - elsif ['warning-mf-bottom-slab-non-zero-foundation-height.xml'].include? hpxml_file + when 'warning-mf-bottom-slab-non-zero-foundation-height.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeSlab args['geometry_foundation_height_above_grade'] = 0.0 args['geometry_attic_type'] = HPXML::AtticTypeBelowApartment - elsif ['warning-slab-non-zero-foundation-height-above-grade.xml'].include? hpxml_file + when 'warning-slab-non-zero-foundation-height-above-grade.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeSlab args['geometry_foundation_height'] = 0.0 args.delete('foundation_wall_insulation_distance_to_bottom') - elsif ['warning-vented-crawlspace-with-wall-and-ceiling-insulation.xml'].include? hpxml_file + when 'warning-vented-crawlspace-with-wall-and-ceiling-insulation.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceVented args['geometry_foundation_height'] = 3.0 args['floor_over_foundation_assembly_r'] = 10 args['foundation_wall_insulation_distance_to_bottom'] = 0.0 args['foundation_wall_assembly_r'] = 10 - elsif ['warning-unvented-crawlspace-with-wall-and-ceiling-insulation.xml'].include? hpxml_file + when 'warning-unvented-crawlspace-with-wall-and-ceiling-insulation.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeCrawlspaceUnvented args['geometry_foundation_height'] = 3.0 args['floor_over_foundation_assembly_r'] = 10 args['foundation_wall_insulation_distance_to_bottom'] = 0.0 args['foundation_wall_assembly_r'] = 10 - elsif ['warning-unconditioned-basement-with-wall-and-ceiling-insulation.xml'].include? hpxml_file + when 'warning-unconditioned-basement-with-wall-and-ceiling-insulation.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeBasementUnconditioned args['floor_over_foundation_assembly_r'] = 10 args['foundation_wall_assembly_r'] = 10 - elsif ['warning-vented-attic-with-floor-and-roof-insulation.xml'].include? hpxml_file + when 'warning-vented-attic-with-floor-and-roof-insulation.xml' args['geometry_attic_type'] = HPXML::AtticTypeVented args['roof_assembly_r'] = 10 args['ducts_supply_location'] = HPXML::LocationAtticVented args['ducts_return_location'] = HPXML::LocationAtticVented - elsif ['warning-unvented-attic-with-floor-and-roof-insulation.xml'].include? hpxml_file + when 'warning-unvented-attic-with-floor-and-roof-insulation.xml' args['geometry_attic_type'] = HPXML::AtticTypeUnvented args['roof_assembly_r'] = 10 - elsif ['warning-conditioned-basement-with-ceiling-insulation.xml'].include? hpxml_file + when 'warning-conditioned-basement-with-ceiling-insulation.xml' args['geometry_foundation_type'] = HPXML::FoundationTypeBasementConditioned args['floor_over_foundation_assembly_r'] = 10 - elsif ['warning-conditioned-attic-with-floor-insulation.xml'].include? hpxml_file + when 'warning-conditioned-attic-with-floor-insulation.xml' args['geometry_unit_num_floors_above_grade'] = 2 args['geometry_attic_type'] = HPXML::AtticTypeConditioned args['ducts_supply_location'] = HPXML::LocationConditionedSpace args['ducts_return_location'] = HPXML::LocationConditionedSpace - elsif ['warning-geothermal-loop-but-no-gshp.xml'].include? hpxml_file + when 'warning-geothermal-loop-but-no-gshp.xml' args['geothermal_loop_configuration'] = HPXML::GeothermalLoopLoopConfigurationVertical end end diff --git a/BuildResidentialScheduleFile/measure.xml b/BuildResidentialScheduleFile/measure.xml index 92ee872119..5ac98b35e0 100644 --- a/BuildResidentialScheduleFile/measure.xml +++ b/BuildResidentialScheduleFile/measure.xml @@ -3,8 +3,8 @@ 3.1 build_residential_schedule_file f770b2db-1a9f-4e99-99a7-7f3161a594b1 - fac68af6-8045-433e-8eca-df53c8274e61 - 2024-09-29T23:07:27Z + 12b68002-26b0-4e2d-bb89-f337226f0ac3 + 2024-12-09T21:40:29Z 03F02484 BuildResidentialScheduleFile Schedule File Builder @@ -229,7 +229,7 @@ schedules.rb rb resource - F53FB2CB + F14B1337 shower_cluster_size_probability.csv diff --git a/BuildResidentialScheduleFile/resources/schedules.rb b/BuildResidentialScheduleFile/resources/schedules.rb index 4720b00c67..540d41b6eb 100644 --- a/BuildResidentialScheduleFile/resources/schedules.rb +++ b/BuildResidentialScheduleFile/resources/schedules.rb @@ -176,7 +176,7 @@ def create_stochastic_schedules(args:, plugload_tv_monthly_multiplier = Schedule.validate_values(schedules_csv_data[SchedulesFile::Columns[:PlugLoadsTV].name]['PlugLoadsTVMonthlyMultipliers'], 12, 'monthly') # American Time Use Survey ceiling_fan_weekday_sch = Schedule.validate_values(default_schedules_csv_data[SchedulesFile::Columns[:CeilingFan].name]['WeekdayScheduleFractions'], 24, 'weekday') # Table C.3(5) of ANSI/RESNET/ICC 301-2022 Addendum C ceiling_fan_weekend_sch = Schedule.validate_values(default_schedules_csv_data[SchedulesFile::Columns[:CeilingFan].name]['WeekendScheduleFractions'], 24, 'weekend') # Table C.3(5) of ANSI/RESNET/ICC 301-2022 Addendum C - ceiling_fan_monthly_multiplier = Schedule.validate_values(Defaults.get_ceiling_fan_months(weather).join(', '), 12, 'monthly') # based on monthly average outdoor temperatures per ANSI/RESNET/ICC 301-2019 + ceiling_fan_monthly_multiplier = Schedule.validate_values(Defaults.get_ceiling_fan_months(weather).join(', '), 12, 'monthly') # based on monthly average outdoor temperatures per ANSI/RESNET/ICC 301 sch = get_building_america_lighting_schedule(args[:time_zone_utc_offset], args[:latitude], args[:longitude], schedules_csv_data) interior_lighting_schedule = [] diff --git a/Changelog.md b/Changelog.md index baf2cef1c9..b2a9029a11 100644 --- a/Changelog.md +++ b/Changelog.md @@ -1,3 +1,18 @@ +## OpenStudio-HPXML v1.10.0 + +__New Features__ + +__Bugfixes__ +- Fixes zero occupants specified for one unit in a whole MF building from being treated like zero occupants for every unit. + +## OpenStudio-HPXML v1.9.1 + +__New Features__ +- Now can be used to obtain ACCA Manual J approval; see the [OpenStudio-HPXML documentation](https://openstudio-hpxml.readthedocs.io/en/latest/intro.html#capabilities). + +__Bugfixes__ +- Fixes Manual J design load calculations for radiant floors. + ## OpenStudio-HPXML v1.9.0 __New Features__ diff --git a/HPXMLtoOpenStudio/measure.rb b/HPXMLtoOpenStudio/measure.rb index b22a139b26..0b68841d76 100644 --- a/HPXMLtoOpenStudio/measure.rb +++ b/HPXMLtoOpenStudio/measure.rb @@ -407,20 +407,6 @@ def init(model, hpxml_bldg, hpxml_header) if hpxml_header.apply_ashrae140_assumptions.nil? hpxml_header.apply_ashrae140_assumptions = false end - - if not hpxml_bldg.building_occupancy.number_of_residents.nil? - # If zero occupants, ensure end uses of interest are zeroed out - if (hpxml_bldg.building_occupancy.number_of_residents == 0) && (not hpxml_header.apply_ashrae140_assumptions) - hpxml_header.unavailable_periods.add(column_name: 'Vacancy', - begin_month: hpxml_header.sim_begin_month, - begin_day: hpxml_header.sim_begin_day, - begin_hour: 0, - end_month: hpxml_header.sim_end_month, - end_day: hpxml_header.sim_end_day, - end_hour: 24, - natvent_availability: HPXML::ScheduleUnavailable) - end - end end end diff --git a/HPXMLtoOpenStudio/measure.xml b/HPXMLtoOpenStudio/measure.xml index acf153b252..bd36e2924f 100644 --- a/HPXMLtoOpenStudio/measure.xml +++ b/HPXMLtoOpenStudio/measure.xml @@ -3,8 +3,8 @@ 3.1 hpxm_lto_openstudio b1543b30-9465-45ff-ba04-1d1f85e763bc - 531c1027-0d75-4343-a8ce-6717ad2b9924 - 2024-11-16T04:46:12Z + f6215039-69e4-4cc9-a911-32c6e2db4759 + 2024-12-17T21:18:42Z D8922A73 HPXMLtoOpenStudio HPXML to OpenStudio Translator @@ -183,13 +183,13 @@ measure.rb rb script - D7F18DFB + CA101CA3 airflow.rb rb resource - 87ABF972 + 06E4A131 battery.rb @@ -213,7 +213,7 @@ constructions.rb rb resource - B3B897EB + 75FBCCF4 data/Xing_okstate_0664D_13659_Table_A-3.csv @@ -327,13 +327,13 @@ defaults.rb rb resource - BCA4AD98 + CADAF9FE energyplus.rb rb resource - 978EE36D + 071927F3 generator.rb @@ -345,19 +345,19 @@ geometry.rb rb resource - 6C147EFE + 58D1C43A hotwater_appliances.rb rb resource - 1EE4897F + 0B205523 hpxml.rb rb resource - 3F71DE61 + 270F2EEB hpxml_schema/HPXML.xsd @@ -375,7 +375,7 @@ hpxml_schematron/EPvalidator.xml xml resource - 94AE43FB + 105F837D hpxml_schematron/iso-schematron.xsd @@ -387,25 +387,25 @@ hvac.rb rb resource - 0DF759D2 + 0F2DDAA0 hvac_sizing.rb rb resource - C6F9CE12 + A3B81222 internal_gains.rb rb resource - 94B4EA05 + 206BE2B6 lighting.rb rb resource - 9B17C563 + 7B7F6D4C location.rb @@ -417,7 +417,7 @@ materials.rb rb resource - 0E69FBCE + 23B30F2C math.rb @@ -441,13 +441,13 @@ misc_loads.rb rb resource - 2DCA8614 + 039B0042 model.rb rb resource - A578B92B + 56A086CD output.rb @@ -465,7 +465,7 @@ pv.rb rb resource - 58457C1A + F8E8DDDD schedule_files/battery.csv @@ -591,7 +591,7 @@ schedules.rb rb resource - 29DAE0EC + 467EB413 simcontrols.rb @@ -615,19 +615,19 @@ utility_bills.rb rb resource - 17195DD4 + C87C3553 version.rb rb resource - CB4D15AB + FB92922A waterheater.rb rb resource - 6F8A18EC + B6FE7ABC weather.rb @@ -639,7 +639,7 @@ xmlhelper.rb rb resource - DA4456A1 + B18A6A72 xmlvalidator.rb @@ -651,7 +651,7 @@ test_airflow.rb rb test - DA1297B3 + 6BDC58E4 test_battery.rb @@ -681,7 +681,7 @@ test_hotwater_appliance.rb rb test - A1E1E023 + 27A0085A test_hvac.rb @@ -699,7 +699,7 @@ test_lighting.rb rb test - 9F945097 + 5300BCE3 test_location.rb @@ -711,7 +711,7 @@ test_miscloads.rb rb test - 974B3838 + 784CC382 test_pv.rb @@ -735,7 +735,7 @@ test_validation.rb rb test - CABF42C5 + 95476C6B test_water_heater.rb diff --git a/HPXMLtoOpenStudio/resources/airflow.rb b/HPXMLtoOpenStudio/resources/airflow.rb index 16b83d830c..8709970c39 100644 --- a/HPXMLtoOpenStudio/resources/airflow.rb +++ b/HPXMLtoOpenStudio/resources/airflow.rb @@ -38,6 +38,10 @@ def self.apply(runner, model, weather, spaces, hpxml_bldg, hpxml_header, schedul elsif f.used_for_seasonal_cooling_load_reduction vent_fans[:whf] << f elsif f.used_for_local_ventilation + if hpxml_bldg.building_occupancy.number_of_residents == 0 + # Operational calculation w/ zero occupants, zero out energy use + continue + end if f.fan_location == HPXML::LocationKitchen vent_fans[:kitchen] << f elsif f.fan_location == HPXML::LocationBath @@ -278,17 +282,18 @@ def self.set_wind_speed_correction(model, hpxml_bldg) site_ap.ashrae_terrain_thickness = 270 site_ap.ashrae_terrain_exponent = 0.14 - if site.site_type == HPXML::SiteTypeRural + case site.site_type + when HPXML::SiteTypeRural site_ap.site_terrain_multiplier = 0.85 site_ap.site_terrain_exponent = 0.20 site_ap.ashrae_site_terrain_thickness = 270 # Flat, open country site_ap.ashrae_site_terrain_exponent = 0.14 # Flat, open country - elsif site.site_type == HPXML::SiteTypeSuburban + when HPXML::SiteTypeSuburban site_ap.site_terrain_multiplier = 0.67 site_ap.site_terrain_exponent = 0.25 site_ap.ashrae_site_terrain_thickness = 370 # Rough, wooded country, suburbs site_ap.ashrae_site_terrain_exponent = 0.22 # Rough, wooded country, suburbs - elsif site.site_type == HPXML::SiteTypeUrban + when HPXML::SiteTypeUrban site_ap.site_terrain_multiplier = 0.47 site_ap.site_terrain_exponent = 0.35 site_ap.ashrae_site_terrain_thickness = 460 # Towns, city outskirts, center of large cities @@ -297,11 +302,12 @@ def self.set_wind_speed_correction(model, hpxml_bldg) # Mapping based on AIM-2 Model by Walker/Wilson # Table 2: Estimates of Shelter Coefficient S_wo for No Flue (flue effect is handled later) - if site.shielding_of_home == HPXML::ShieldingNormal + case site.shielding_of_home + when HPXML::ShieldingNormal site_ap.aim2_shelter_coeff = 0.50 # Class 4: "Very heavy shielding, many large obstructions within one house height" - elsif site.shielding_of_home == HPXML::ShieldingExposed + when HPXML::ShieldingExposed site_ap.aim2_shelter_coeff = 0.90 # Class 2: "Light local shielding with few obstructions within two house heights" - elsif site.shielding_of_home == HPXML::ShieldingWellShielded + when HPXML::ShieldingWellShielded site_ap.aim2_shelter_coeff = 0.30 # Class 5: "Complete shielding, with large buildings immediately adjacent" end @@ -528,7 +534,7 @@ def self.apply_natural_ventilation_and_whole_house_fan(runner, model, spaces, hp neutral_level = 0.5 hor_lk_frac = 0.0 c_w, c_s = calc_wind_stack_coeffs(hpxml_bldg, hor_lk_frac, neutral_level, conditioned_space, infil_values[:height]) - max_oa_hr = 0.0115 # From ANSI 301-2022 + max_oa_hr = 0.0115 # From ANSI/RESNET/ICC 301-2022 # Program vent_program = Model.add_ems_program( @@ -586,12 +592,19 @@ def self.apply_natural_ventilation_and_whole_house_fan(runner, model, spaces, hp vent_program.addLine(' Set Qwhf = WHF_Flow*Adj') vent_program.addLine(" Set #{cond_to_zone_flow_rate_actuator.name} = WHF_Flow*Adj") unless whf_zone.nil? vent_program.addLine(' Set WHF_W = 0') - vent_fans[:whf].each do |vent_whf| - vent_program.addLine(" Set WHF_W = WHF_W + #{vent_whf.fan_power} * #{whf_avail_sensors[vent_whf.id].name}") + if hpxml_bldg.building_occupancy.number_of_residents != 0 # If operational calculation w/ zero occupants, zero out whole house fan + vent_fans[:whf].each do |vent_whf| + vent_program.addLine(" Set WHF_W = WHF_W + #{vent_whf.fan_power} * #{whf_avail_sensors[vent_whf.id].name}") + end end vent_program.addLine(" Set #{whf_elec_actuator.name} = WHF_W*Adj") vent_program.addLine(' ElseIf (NVavail > 0)') # Natural ventilation - vent_program.addLine(" Set NVArea = #{UnitConversions.convert(area, 'ft^2', 'cm^2')}") + if hpxml_bldg.building_occupancy.number_of_residents == 0 + # Operational calculation w/ zero occupants, zero out natural ventilation + vent_program.addLine(' Set NVArea = 0') + else + vent_program.addLine(" Set NVArea = #{UnitConversions.convert(area, 'ft^2', 'cm^2')}") + end vent_program.addLine(" Set Cs = #{UnitConversions.convert(c_s, 'ft^2/(s^2*R)', 'L^2/(s^2*cm^4*K)')}") vent_program.addLine(" Set Cw = #{c_w * 0.01}") vent_program.addLine(' Set Tdiff = Tin-Tout') @@ -854,11 +867,12 @@ def self.check_duct_leakage(runner, hpxml_bldg) if hvac_distribution.ducts.count { |d| !HPXML::conditioned_locations_this_unit.include?(d.duct_location) } == 0 # If ducts completely in conditioned space, issue warning if duct leakage to outside above a certain threshold (e.g., 5%) issue_warning = false - if units == HPXML::UnitsCFM25 + case units + when HPXML::UnitsCFM25 issue_warning = true if sum_lto > 0.04 * cfa - elsif units == HPXML::UnitsCFM50 + when HPXML::UnitsCFM50 issue_warning = true if sum_lto > 0.06 * cfa - elsif units == HPXML::UnitsPercent + when HPXML::UnitsPercent issue_warning = true if sum_lto > 0.05 end next unless issue_warning @@ -867,11 +881,12 @@ def self.check_duct_leakage(runner, hpxml_bldg) else # If ducts in unconditioned space, issue warning if duct leakage to outside above a certain threshold (e.g., 40%) issue_warning = false - if units == HPXML::UnitsCFM25 + case units + when HPXML::UnitsCFM25 issue_warning = true if sum_lto >= 0.32 * cfa - elsif units == HPXML::UnitsCFM50 + when HPXML::UnitsCFM50 issue_warning = true if sum_lto >= 0.48 * cfa - elsif units == HPXML::UnitsPercent + when HPXML::UnitsPercent issue_warning = true if sum_lto >= 0.4 end next unless issue_warning @@ -1455,7 +1470,7 @@ def self.apply_duct_location(model, spaces, hpxml_bldg, ducts, object, i, duct_l duct_subroutine.addLine(' Set SupLatLkToDZ = sup_lk_mfr*h_fg*(AH_Wout-DZ_W)') # W duct_subroutine.addLine(' Set SupSensLkToDZ = SupTotLkToDZ-SupLatLkToDZ') # W - # Handle duct leakage imbalance induced infiltration (ANSI 301-2022 Addendum C Table 4.2.2(1c) + # Handle duct leakage imbalance induced infiltration (ANSI/RESNET/ICC 301-2022 Addendum C Table 4.2.2(1c) leakage_supply = leakage_fracs[HPXML::DuctTypeSupply].to_f + leakage_cfm25s[HPXML::DuctTypeSupply].to_f leakage_return = leakage_fracs[HPXML::DuctTypeReturn].to_f + leakage_cfm25s[HPXML::DuctTypeReturn].to_f if leakage_supply == leakage_return @@ -1984,30 +1999,32 @@ def self.apply_cfis(runner, infil_program, vent_mech_fans, cfis_data, cfis_fan_a infil_program.addLine("Set f_operation = #{[vent_mech.hours_in_operation / 24.0, 0.0001].max}") # Operation, fraction of hour infil_program.addLine("Set oa_cfm_ah = #{UnitConversions.convert(vent_mech.oa_unit_flow_rate, 'cfm', 'm^3/s')}") - if vent_mech.cfis_addtl_runtime_operating_mode == HPXML::CFISModeSupplementalFan + case vent_mech.cfis_addtl_runtime_operating_mode + when HPXML::CFISModeSupplementalFan if vent_mech.cfis_supplemental_fan.oa_unit_flow_rate < vent_mech.average_unit_flow_rate runner.registerWarning("CFIS supplemental fan '#{vent_mech.cfis_supplemental_fan.id}' is undersized (#{vent_mech.cfis_supplemental_fan.oa_unit_flow_rate} cfm) compared to the target hourly ventilation rate (#{vent_mech.average_unit_flow_rate} cfm).") end infil_program.addLine("Set suppl_fan_w = #{vent_mech.cfis_supplemental_fan.unit_fan_power}") # W, supplemental fan power infil_program.addLine("Set oa_cfm_suppl = #{UnitConversions.convert(vent_mech.cfis_supplemental_fan.oa_unit_flow_rate, 'cfm', 'm^3/s')}") - elsif vent_mech.cfis_addtl_runtime_operating_mode == HPXML::CFISModeAirHandler + when HPXML::CFISModeAirHandler infil_program.addLine("Set ah_fan_w = #{vent_mech.unit_fan_power}") # W, air handler fan power end infil_program.addLine("Set has_outdoor_air_control = #{vent_mech.cfis_has_outdoor_air_control ? 1 : 0}") - if vent_mech.cfis_control_type == HPXML::CFISControlTypeTimer + case vent_mech.cfis_control_type + when HPXML::CFISControlTypeTimer # Ventilation occurs at fixed intervals regardless of HVAC operation # Calculate outdoor air ventilation infil_program.addLine('Set QWHV_cfis_sup = QWHV_cfis_sup + (oa_cfm_ah * f_operation)') - # Calculate fraction of the timestep with ventilation only mode runtime per ANSI 301-2022 Addendum E + # Calculate fraction of the timestep with ventilation only mode runtime per ANSI/RESNET/ICC 301-2022 Addendum E infil_program.addLine("Set #{f_vent_only_mode_var.name} = f_operation * (1 - fan_rtf_hvac)") # Calculate additional fan energy infil_program.addLine("Set #{cfis_fan_actuator.name} = #{cfis_fan_actuator.name} + (ah_fan_w * #{f_vent_only_mode_var.name})") - elsif vent_mech.cfis_control_type == HPXML::CFISControlTypeOptimized + when HPXML::CFISControlTypeOptimized # Ventilation optimized to make use of HVAC operation infil_program.addLine("Set #{f_vent_only_mode_var.name} = 0") @@ -2040,9 +2057,10 @@ def self.apply_cfis(runner, infil_program, vent_mech_fans, cfis_data, cfis_fan_a infil_program.addLine(' If hr_oa_cfm_during_hvac > 0') infil_program.addLine(' Set f_open_damper_ah = (hr_oa_cfm_during_hvac / hr_oa_cfm_during_hvac_avail) * fan_rtf_hvac') infil_program.addLine(" Set #{cfis_suppl_fan_actuator.name} = #{cfis_suppl_fan_actuator.name} + (suppl_fan_w * f_open_damper_ah)") - if vent_mech.cfis_supplemental_fan.fan_type == HPXML::MechVentTypeSupply + case vent_mech.cfis_supplemental_fan.fan_type + when HPXML::MechVentTypeSupply infil_program.addLine(' Set QWHV_cfis_suppl_sup = QWHV_cfis_suppl_sup + (f_open_damper_ah * oa_cfm_suppl)') - elsif vent_mech.cfis_supplemental_fan.fan_type == HPXML::MechVentTypeExhaust + when HPXML::MechVentTypeExhaust infil_program.addLine(' Set QWHV_cfis_suppl_exh = QWHV_cfis_suppl_exh + (f_open_damper_ah * oa_cfm_suppl)') end infil_program.addLine(' EndIf') @@ -2052,16 +2070,18 @@ def self.apply_cfis(runner, infil_program, vent_mech_fans, cfis_data, cfis_fan_a infil_program.addLine(' Set hr_oa_cfm_addtl_needed = hr_oa_cfm_needed - hr_oa_cfm_during_hvac') # Calculate hourly-average outdoor air ventilation that can be brought in during subsequent timesteps of the hour if needed - if vent_mech.cfis_addtl_runtime_operating_mode == HPXML::CFISModeAirHandler + case vent_mech.cfis_addtl_runtime_operating_mode + when HPXML::CFISModeAirHandler infil_program.addLine(' Set hr_oa_cfm_addtl_needed = hr_oa_cfm_addtl_needed - (((60 - Minute) / 60) * oa_cfm_ah)') - elsif vent_mech.cfis_addtl_runtime_operating_mode == HPXML::CFISModeSupplementalFan + when HPXML::CFISModeSupplementalFan infil_program.addLine(' Set hr_oa_cfm_addtl_needed = hr_oa_cfm_addtl_needed - (((60 - Minute) / 60) * oa_cfm_suppl)') - elsif vent_mech.cfis_addtl_runtime_operating_mode == HPXML::CFISModeNone + when HPXML::CFISModeNone infil_program.addLine(' Set hr_oa_cfm_addtl_needed = 0') end infil_program.addLine(' If hr_oa_cfm_addtl_needed > 0') - if vent_mech.cfis_addtl_runtime_operating_mode == HPXML::CFISModeAirHandler + case vent_mech.cfis_addtl_runtime_operating_mode + when HPXML::CFISModeAirHandler # Air handler meets additional runtime requirement # Calculate hourly-average available outdoor air ventilation during non-HVAC runtime @@ -2078,7 +2098,7 @@ def self.apply_cfis(runner, infil_program, vent_mech_fans, cfis_data, cfis_fan_a infil_program.addLine(" Set #{cfis_fan_actuator.name} = #{cfis_fan_actuator.name} + (ah_fan_w * #{f_vent_only_mode_var.name})") infil_program.addLine(' EndIf') - elsif vent_mech.cfis_addtl_runtime_operating_mode == HPXML::CFISModeSupplementalFan + when HPXML::CFISModeSupplementalFan # Supplemental fan meets additional runtime requirement # Calculate hourly-average available outdoor air ventilation during non-HVAC runtime @@ -2086,9 +2106,10 @@ def self.apply_cfis(runner, infil_program, vent_mech_fans, cfis_data, cfis_fan_a # Calculate hourly-average actual outdoor air ventilation brought in during non-HVAC runtime infil_program.addLine(' Set hr_oa_cfm_during_non_hvac = @Min hr_oa_cfm_during_non_hvac_avail hr_oa_cfm_addtl_needed') - if vent_mech.cfis_supplemental_fan.fan_type == HPXML::MechVentTypeSupply + case vent_mech.cfis_supplemental_fan.fan_type + when HPXML::MechVentTypeSupply infil_program.addLine(' Set QWHV_cfis_suppl_sup = QWHV_cfis_suppl_sup + (hr_oa_cfm_during_non_hvac / ZoneTimestep)') - elsif vent_mech.cfis_supplemental_fan.fan_type == HPXML::MechVentTypeExhaust + when HPXML::MechVentTypeExhaust infil_program.addLine(' Set QWHV_cfis_suppl_exh = QWHV_cfis_suppl_exh + (hr_oa_cfm_during_non_hvac / ZoneTimestep)') end infil_program.addLine(" Set #{sum_oa_cfm_var.name} = #{sum_oa_cfm_var.name} + hr_oa_cfm_during_non_hvac") @@ -2288,6 +2309,7 @@ def self.apply_infiltration_adjustment_to_conditioned(runner, model, spaces, hpx clothes_dryer_in_cond_space = hpxml_bldg.clothes_dryers.empty? ? true : HPXML::conditioned_locations_this_unit.include?(hpxml_bldg.clothes_dryers[0].location) vented_dryers = hpxml_bldg.clothes_dryers.select { |cd| cd.is_vented && cd.vented_flow_rate.to_f > 0 } vented_dryers.each_with_index do |vented_dryer, index| + next if hpxml_bldg.building_occupancy.number_of_residents == 0 # Operational calculation w/ zero occupants, zero out energy use next unless clothes_dryer_in_cond_space # Infiltration impact @@ -2905,11 +2927,11 @@ def self.get_mech_vent_qtot_cfm(nbeds, cfa) # @param is_balanced [TODO] TODO # @param frac_imbal [TODO] TODO # @param a_ext [TODO] TODO - # @param bldg_type [TODO] TODO + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @param eri_version [String] Version of the ANSI/RESNET/ICC 301 Standard to use for equations/assumptions # @param hours_in_operation [TODO] TODO # @return [TODO] TODO - def self.get_mech_vent_qfan_cfm(q_tot, q_inf, is_balanced, frac_imbal, a_ext, bldg_type, eri_version, hours_in_operation) + def self.get_mech_vent_qfan_cfm(q_tot, q_inf, is_balanced, frac_imbal, a_ext, unit_type, eri_version, hours_in_operation) q_inf_eff = q_inf * a_ext if Constants::ERIVersions.index(eri_version) >= Constants::ERIVersions.index('2022') if frac_imbal == 0 @@ -2930,7 +2952,7 @@ def self.get_mech_vent_qfan_cfm(q_tot, q_inf, is_balanced, frac_imbal, a_ext, bl end q_fan = q_tot - phi * q_inf_eff else - if [HPXML::ResidentialTypeApartment, HPXML::ResidentialTypeSFA].include? bldg_type + if [HPXML::ResidentialTypeApartment, HPXML::ResidentialTypeSFA].include? unit_type # No infiltration credit for attached/multifamily return q_tot end @@ -3057,11 +3079,12 @@ def self.create_ducts(model, hvac_distribution, spaces) duct_leakage_value = leakage_to_outside[duct_side][0] duct_leakage_units = leakage_to_outside[duct_side][1] - if duct_leakage_units == HPXML::UnitsCFM25 + case duct_leakage_units + when HPXML::UnitsCFM25 duct_leakage_cfm25 = duct_leakage_value - elsif duct_leakage_units == HPXML::UnitsCFM50 + when HPXML::UnitsCFM50 duct_leakage_cfm50 = duct_leakage_value - elsif duct_leakage_units == HPXML::UnitsPercent + when HPXML::UnitsPercent duct_leakage_frac = duct_leakage_value else fail "#{duct_side.capitalize} ducts exist but leakage was not specified for distribution system '#{hvac_distribution.id}'." diff --git a/HPXMLtoOpenStudio/resources/constructions.rb b/HPXMLtoOpenStudio/resources/constructions.rb index 36c34aca50..252409cd60 100644 --- a/HPXMLtoOpenStudio/resources/constructions.rb +++ b/HPXMLtoOpenStudio/resources/constructions.rb @@ -1544,13 +1544,15 @@ def self.apply_furniture(model, furniture_mass, spaces) furnSolarAbsorptance = 0.6 furnSpecHeat = mat.cp furnDensity = mat.rho - if location == HPXML::LocationConditionedSpace + + case location + when HPXML::LocationConditionedSpace furnAreaFraction = furniture_mass.area_fraction furnMass = mass_lb_per_sqft - elsif location == HPXML::LocationBasementUnconditioned + when HPXML::LocationBasementUnconditioned furnAreaFraction = 0.4 furnMass = mass_lb_per_sqft - elsif location == HPXML::LocationGarage + when HPXML::LocationGarage furnAreaFraction = 0.1 furnMass = 2.0 end @@ -1686,11 +1688,14 @@ def self.get_wall_color_and_solar_absorptance_map def self.get_gap_factor(install_grade, framing_factor, cavity_r) if cavity_r <= 0 return 0 # Gap factor only applies when there is cavity insulation - elsif install_grade == 1 + end + + case install_grade + when 1 return 0 - elsif install_grade == 2 + when 2 return 0.02 * (1 - framing_factor) - elsif install_grade == 3 + when 3 return 0.05 * (1 - framing_factor) end @@ -1920,7 +1925,8 @@ def self.apply_kiva_initial_temperature(foundation, weather, hpxml_bldg, hpxml_h else # Space temperature assumptions from ASHRAE 152 - Duct Efficiency Calculations.xls, Zone temperatures ground_temp = weather.data.ShallowGroundMonthlyTemps[sim_begin_month - 1] - if interior_adjacent_to == HPXML::LocationBasementUnconditioned + case interior_adjacent_to + when HPXML::LocationBasementUnconditioned if foundation_ceiling_insulated # Insulated ceiling: 75% ground, 25% outdoor, 0% indoor ground_weight, outdoor_weight, indoor_weight = 0.75, 0.25, 0.0 @@ -1932,7 +1938,7 @@ def self.apply_kiva_initial_temperature(foundation, weather, hpxml_bldg, hpxml_h ground_weight, outdoor_weight, indoor_weight = 0.5, 0.2, 0.3 end initial_temp = outdoor_temp * outdoor_weight + ground_temp * ground_weight + indoor_weight * indoor_temp - elsif interior_adjacent_to == HPXML::LocationCrawlspaceVented + when HPXML::LocationCrawlspaceVented if foundation_ceiling_insulated # Insulated ceiling: 90% outdoor, 10% indoor outdoor_weight, indoor_weight = 0.9, 0.1 @@ -1944,7 +1950,7 @@ def self.apply_kiva_initial_temperature(foundation, weather, hpxml_bldg, hpxml_h outdoor_weight, indoor_weight = 0.5, 0.5 end initial_temp = outdoor_temp * outdoor_weight + indoor_weight * indoor_temp - elsif interior_adjacent_to == HPXML::LocationCrawlspaceUnvented + when HPXML::LocationCrawlspaceUnvented if foundation_ceiling_insulated # Insulated ceiling: 85% outdoor, 15% indoor outdoor_weight, indoor_weight = 0.85, 0.15 @@ -1956,7 +1962,7 @@ def self.apply_kiva_initial_temperature(foundation, weather, hpxml_bldg, hpxml_h outdoor_weight, indoor_weight = 0.4, 0.6 end initial_temp = outdoor_temp * outdoor_weight + indoor_weight * indoor_temp - elsif interior_adjacent_to == HPXML::LocationGarage + when HPXML::LocationGarage initial_temp = outdoor_temp + 11.0 else fail "Unhandled space: #{interior_adjacent_to}" @@ -2036,14 +2042,14 @@ def self.apply_window_skylight_shading(model, window_or_skylight, sub_surface, s esf_winter = window_or_skylight.exterior_shading_factor_winter.nil? ? 1.0 : window_or_skylight.exterior_shading_factor_winter if window_or_skylight.is_a? HPXML::Window # These inputs currently only pertain to windows (not skylights) - if [HPXML::ExteriorShadingTypeExternalOverhangs, - HPXML::ExteriorShadingTypeAwnings].include? window_or_skylight.exterior_shading_type + case window_or_skylight.exterior_shading_type + when HPXML::ExteriorShadingTypeExternalOverhangs, HPXML::ExteriorShadingTypeAwnings if window_or_skylight.overhangs_depth.to_f > 0 # Explicitly modeling the overhangs, so don't double count the shading effect esf_summer = 1.0 esf_winter = 1.0 end - elsif [HPXML::ExteriorShadingTypeBuilding].include? window_or_skylight.exterior_shading_type + when HPXML::ExteriorShadingTypeBuilding if hpxml_bldg.neighbor_buildings.size > 0 # Explicitly modeling neighboring building, so don't double count the shading effect esf_summer = 1.0 @@ -2184,7 +2190,8 @@ def self.apply_wall_construction(runner, fallback_mat_int_finish = Material.InteriorFinishMaterial(mat_int_finish.name, 0.1) # Try thin material end - if wall_type == HPXML::WallTypeWoodStud + case wall_type + when HPXML::WallTypeWoodStud install_grade = 1 cavity_filled = true @@ -2215,7 +2222,7 @@ def self.apply_wall_construction(runner, radiant_barrier_grade, solar_absorptance, emittance) - elsif wall_type == HPXML::WallTypeSteelStud + when HPXML::WallTypeSteelStud install_grade = 1 cavity_filled = true corr_factor = 0.45 @@ -2236,7 +2243,7 @@ def self.apply_wall_construction(runner, constr_set.osb_thick_in, constr_set.rigid_r, constr_set.mat_ext_finish, has_radiant_barrier, inside_film, outside_film, radiant_barrier_grade, solar_absorptance, emittance) - elsif wall_type == HPXML::WallTypeDoubleWoodStud + when HPXML::WallTypeDoubleWoodStud install_grade = 1 is_staggered = false @@ -2255,7 +2262,7 @@ def self.apply_wall_construction(runner, has_radiant_barrier, inside_film, outside_film, radiant_barrier_grade, solar_absorptance, emittance) - elsif wall_type == HPXML::WallTypeCMU + when HPXML::WallTypeCMU density = 119.0 # lb/ft^3 furring_r = 0 furring_cavity_depth_in = 0 # in @@ -2274,7 +2281,7 @@ def self.apply_wall_construction(runner, constr_set.mat_int_finish, constr_set.osb_thick_in, rigid_r, constr_set.mat_ext_finish, has_radiant_barrier, inside_film, outside_film, radiant_barrier_grade, solar_absorptance, emittance) - elsif wall_type == HPXML::WallTypeSIP + when HPXML::WallTypeSIP sheathing_thick_in = 0.44 constr_sets = [ @@ -2290,7 +2297,7 @@ def self.apply_wall_construction(runner, constr_set.osb_thick_in, constr_set.rigid_r, constr_set.mat_ext_finish, has_radiant_barrier, inside_film, outside_film, radiant_barrier_grade, solar_absorptance, emittance) - elsif wall_type == HPXML::WallTypeICF + when HPXML::WallTypeICF constr_sets = [ ICFConstructionSet.new(2.0, 4.0, 0.08, 0.0, 0.5, mat_int_finish, mat_ext_finish), # ICF w/4" concrete and 2" rigid ins layers ICFConstructionSet.new(1.0, 1.0, 0.01, 0.0, 0.0, fallback_mat_int_finish, fallback_mat_ext_finish), # Fallback @@ -2305,7 +2312,7 @@ def self.apply_wall_construction(runner, has_radiant_barrier, inside_film, outside_film, radiant_barrier_grade, solar_absorptance, emittance) - elsif [HPXML::WallTypeConcrete, HPXML::WallTypeBrick, HPXML::WallTypeAdobe, HPXML::WallTypeStrawBale, HPXML::WallTypeStone, HPXML::WallTypeLog].include? wall_type + when HPXML::WallTypeConcrete, HPXML::WallTypeBrick, HPXML::WallTypeAdobe, HPXML::WallTypeStrawBale, HPXML::WallTypeStone, HPXML::WallTypeLog constr_sets = [ GenericConstructionSet.new(10.0, 0.5, mat_int_finish, mat_ext_finish), # w/R-10 rigid GenericConstructionSet.new(0.0, 0.5, mat_int_finish, mat_ext_finish), # Standard @@ -2313,22 +2320,23 @@ def self.apply_wall_construction(runner, ] match, constr_set, layer_r = pick_generic_construction_set(assembly_r, constr_sets, inside_film, outside_film) - if wall_type == HPXML::WallTypeConcrete + case wall_type + when HPXML::WallTypeConcrete thick_in = 6.0 base_mat = BaseMaterial.Concrete - elsif wall_type == HPXML::WallTypeBrick + when HPXML::WallTypeBrick thick_in = 8.0 base_mat = BaseMaterial.Brick - elsif wall_type == HPXML::WallTypeAdobe + when HPXML::WallTypeAdobe thick_in = 10.0 base_mat = BaseMaterial.Soil(12.0) - elsif wall_type == HPXML::WallTypeStrawBale + when HPXML::WallTypeStrawBale thick_in = 23.0 base_mat = BaseMaterial.StrawBale - elsif wall_type == HPXML::WallTypeStone + when HPXML::WallTypeStone thick_in = 6.0 base_mat = BaseMaterial.Stone - elsif wall_type == HPXML::WallTypeLog + when HPXML::WallTypeLog thick_in = 6.0 base_mat = BaseMaterial.Wood end @@ -2385,7 +2393,8 @@ def self.apply_floor_ceiling_construction(runner, model, surface, floor_id, floo end osb_thick_in = (is_ceiling ? 0.0 : 0.75) - if floor_type == HPXML::FloorTypeWoodFrame + case floor_type + when HPXML::FloorTypeWoodFrame install_grade = 1 constr_sets = [ WoodStudConstructionSet.new(Material.Stud2x6, 0.10, 50.0, osb_thick_in, mat_int_finish_or_covering, nil), # 2x6, 24" o.c. + R50 @@ -2406,7 +2415,7 @@ def self.apply_floor_ceiling_construction(runner, model, surface, floor_id, floo constr_set.osb_thick_in, constr_set.rigid_r, constr_int_finish_or_covering, has_radiant_barrier, inside_film, outside_film, radiant_barrier_grade) - elsif floor_type == HPXML::FloorTypeSteelFrame + when HPXML::FloorTypeSteelFrame install_grade = 1 corr_factor = 0.45 osb_thick_in = (is_ceiling ? 0.0 : 0.75) @@ -2428,7 +2437,7 @@ def self.apply_floor_ceiling_construction(runner, model, surface, floor_id, floo constr_set.osb_thick_in, constr_set.rigid_r, constr_int_finish_or_covering, has_radiant_barrier, inside_film, outside_film, radiant_barrier_grade) - elsif floor_type == HPXML::FloorTypeSIP + when HPXML::FloorTypeSIP constr_sets = [ SIPConstructionSet.new(16.0, 0.08, 0.0, 0.0, osb_thick_in, mat_int_finish_or_covering, nil), # 16" SIP core SIPConstructionSet.new(12.0, 0.08, 0.0, 0.0, osb_thick_in, mat_int_finish_or_covering, nil), # 12" SIP core @@ -2441,7 +2450,7 @@ def self.apply_floor_ceiling_construction(runner, model, surface, floor_id, floo cavity_r, constr_set.thick_in, constr_set.framing_factor, constr_set.mat_int_finish, constr_set.osb_thick_in, constr_set.rigid_r, constr_set.mat_ext_finish, has_radiant_barrier, inside_film, outside_film, radiant_barrier_grade) - elsif floor_type == HPXML::FloorTypeConcrete + when HPXML::FloorTypeConcrete constr_sets = [ GenericConstructionSet.new(20.0, osb_thick_in, mat_int_finish_or_covering, nil), # w/R-20 rigid GenericConstructionSet.new(10.0, osb_thick_in, mat_int_finish_or_covering, nil), # w/R-10 rigid diff --git a/HPXMLtoOpenStudio/resources/defaults.rb b/HPXMLtoOpenStudio/resources/defaults.rb index 5f84296aab..841bc497d1 100644 --- a/HPXMLtoOpenStudio/resources/defaults.rb +++ b/HPXMLtoOpenStudio/resources/defaults.rb @@ -24,7 +24,7 @@ module Defaults # @param hpxml_bldg [HPXML::Building] HPXML Building object representing an individual dwelling unit # @param weather [WeatherFile] Weather object containing EPW information # @param schedules_file [SchedulesFile] SchedulesFile wrapper class instance of detailed schedule files - # @param convert_shared_systems [Boolean] Whether to convert shared systems to equivalent in-unit systems per ANSI 301 + # @param convert_shared_systems [Boolean] Whether to convert shared systems to equivalent in-unit systems per ANSI/RESNET/ICC 301 # @return [Array] Maps of HPXML::Zones => DesignLoadValues object, HPXML::Spaces => DesignLoadValues object def self.apply(runner, hpxml, hpxml_bldg, weather, schedules_file: nil, convert_shared_systems: true) eri_version = hpxml.header.eri_calculation_version @@ -41,7 +41,7 @@ def self.apply(runner, hpxml, hpxml_bldg, weather, schedules_file: nil, convert_ end # Check for presence of fuels once - has_fuel = hpxml_bldg.has_fuels(hpxml.to_doc) + has_fuel = hpxml_bldg.has_fuels() add_zones_spaces_if_needed(hpxml_bldg, unit_num) @@ -652,7 +652,8 @@ def self.apply_building(hpxml_bldg, weather) # Conductivity/diffusivity values come from https://www.ncbi.nlm.nih.gov/pmc/articles/PMC4813881 (with the exception of "unknown") if hpxml_bldg.site.ground_conductivity.nil? && hpxml_bldg.site.ground_diffusivity.nil? - if hpxml_bldg.site.soil_type == HPXML::SiteSoilTypeSand + case hpxml_bldg.site.soil_type + when HPXML::SiteSoilTypeSand if hpxml_bldg.site.moisture_type == HPXML::SiteSoilMoistureTypeDry hpxml_bldg.site.ground_conductivity = 0.2311 # Btu/hr-ft-F hpxml_bldg.site.ground_diffusivity = 0.0097 # ft^2/hr @@ -665,39 +666,40 @@ def self.apply_building(hpxml_bldg, weather) end hpxml_bldg.site.ground_conductivity_isdefaulted = true hpxml_bldg.site.ground_diffusivity_isdefaulted = true - elsif hpxml_bldg.site.soil_type == HPXML::SiteSoilTypeSilt || hpxml_bldg.site.soil_type == HPXML::SiteSoilTypeClay - if hpxml_bldg.site.moisture_type == HPXML::SiteSoilMoistureTypeDry + when HPXML::SiteSoilTypeSilt, HPXML::SiteSoilTypeClay + case hpxml_bldg.site.moisture_type + when HPXML::SiteSoilMoistureTypeDry hpxml_bldg.site.ground_conductivity = 0.2889 # Btu/hr-ft-F hpxml_bldg.site.ground_diffusivity = 0.0120 # ft^2/hr - elsif hpxml_bldg.site.moisture_type == HPXML::SiteSoilMoistureTypeWet + when HPXML::SiteSoilMoistureTypeWet hpxml_bldg.site.ground_conductivity = 0.9821 # Btu/hr-ft-F hpxml_bldg.site.ground_diffusivity = 0.0194 # ft^2/hr - elsif hpxml_bldg.site.moisture_type == HPXML::SiteSoilMoistureTypeMixed + when HPXML::SiteSoilMoistureTypeMixed hpxml_bldg.site.ground_conductivity = ((0.2889 + 0.9821) / 2.0).round(4) # Btu/hr-ft-F hpxml_bldg.site.ground_diffusivity = ((0.0120 + 0.0194) / 2.0).round(4) # ft^2/hr end hpxml_bldg.site.ground_conductivity_isdefaulted = true hpxml_bldg.site.ground_diffusivity_isdefaulted = true - elsif hpxml_bldg.site.soil_type == HPXML::SiteSoilTypeLoam + when HPXML::SiteSoilTypeLoam hpxml_bldg.site.ground_conductivity = 1.2132 # Btu/hr-ft-F hpxml_bldg.site.ground_diffusivity = 0.0353 # ft^2/hr - hpxml_bldg.site.ground_conductivity_isdefaulted = true hpxml_bldg.site.ground_diffusivity_isdefaulted = true - elsif hpxml_bldg.site.soil_type == HPXML::SiteSoilTypeGravel - if hpxml_bldg.site.moisture_type == HPXML::SiteSoilMoistureTypeDry + when HPXML::SiteSoilTypeGravel + case hpxml_bldg.site.moisture_type + when HPXML::SiteSoilMoistureTypeDry hpxml_bldg.site.ground_conductivity = 0.2311 # Btu/hr-ft-F hpxml_bldg.site.ground_diffusivity = 0.0097 # ft^2/hr - elsif hpxml_bldg.site.moisture_type == HPXML::SiteSoilMoistureTypeWet + when HPXML::SiteSoilMoistureTypeWet hpxml_bldg.site.ground_conductivity = 1.0399 # Btu/hr-ft-F hpxml_bldg.site.ground_diffusivity = 0.0291 # ft^2/hr - elsif hpxml_bldg.site.moisture_type == HPXML::SiteSoilMoistureTypeMixed + when HPXML::SiteSoilMoistureTypeMixed hpxml_bldg.site.ground_conductivity = ((0.2311 + 1.0399) / 2.0).round(4) # Btu/hr-ft-F hpxml_bldg.site.ground_diffusivity = ((0.0097 + 0.0291) / 2.0).round(4) # ft^2/hr end hpxml_bldg.site.ground_conductivity_isdefaulted = true hpxml_bldg.site.ground_diffusivity_isdefaulted = true - elsif hpxml_bldg.site.soil_type == HPXML::SiteSoilTypeUnknown + when HPXML::SiteSoilTypeUnknown hpxml_bldg.site.ground_conductivity = 1.0 # ANSI/RESNET/ICC 301-2022 Addendum C hpxml_bldg.site.ground_diffusivity = 0.0208 hpxml_bldg.site.ground_conductivity_isdefaulted = true @@ -827,13 +829,6 @@ def self.apply_neighbor_buildings(hpxml_bldg) # @param schedules_file [SchedulesFile] SchedulesFile wrapper class instance of detailed schedule files # @return [nil] def self.apply_building_occupancy(hpxml_bldg, schedules_file) - if not hpxml_bldg.building_occupancy.number_of_residents.nil? - # Set equivalent number of bedrooms for operational calculation; this is an adjustment on - # ANSI 301 or Building America equations, which are based on number of bedrooms. - hpxml_bldg.building_construction.additional_properties.equivalent_number_of_bedrooms = get_equivalent_nbeds_for_operational_calculation(hpxml_bldg) - else - hpxml_bldg.building_construction.additional_properties.equivalent_number_of_bedrooms = hpxml_bldg.building_construction.number_of_bedrooms - end schedules_file_includes_occupants = (schedules_file.nil? ? false : schedules_file.includes_col_name(SchedulesFile::Columns[:Occupants].name)) if hpxml_bldg.building_occupancy.weekday_fractions.nil? && !schedules_file_includes_occupants hpxml_bldg.building_occupancy.weekday_fractions = @default_schedules_csv_data[SchedulesFile::Columns[:Occupants].name]['WeekdayScheduleFractions'] @@ -1826,7 +1821,7 @@ def self.apply_furniture_mass(hpxml_bldg) # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings # @param hpxml_bldg [HPXML::Building] HPXML Building object representing an individual dwelling unit # @param weather [WeatherFile] Weather object containing EPW information - # @param convert_shared_systems [Boolean] Whether to convert shared systems to equivalent in-unit systems per ANSI 301 + # @param convert_shared_systems [Boolean] Whether to convert shared systems to equivalent in-unit systems per ANSI/RESNET/ICC 301 # @param unit_num [Integer] Dwelling unit number # @return [nil] def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_num) @@ -2008,20 +2003,21 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu hpxml_bldg.cooling_systems.each do |cooling_system| next unless cooling_system.cooling_shr.nil? - if cooling_system.cooling_system_type == HPXML::HVACTypeCentralAirConditioner - if cooling_system.compressor_type == HPXML::HVACCompressorTypeSingleStage + case cooling_system.cooling_system_type + when HPXML::HVACTypeCentralAirConditioner + case cooling_system.compressor_type + when HPXML::HVACCompressorTypeSingleStage cooling_system.cooling_shr = 0.73 - elsif cooling_system.compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage cooling_system.cooling_shr = 0.73 - elsif cooling_system.compressor_type == HPXML::HVACCompressorTypeVariableSpeed + when HPXML::HVACCompressorTypeVariableSpeed cooling_system.cooling_shr = 0.78 end cooling_system.cooling_shr_isdefaulted = true - elsif cooling_system.cooling_system_type == HPXML::HVACTypeRoomAirConditioner || - cooling_system.cooling_system_type == HPXML::HVACTypePTAC + when HPXML::HVACTypeRoomAirConditioner, HPXML::HVACTypePTAC cooling_system.cooling_shr = 0.65 cooling_system.cooling_shr_isdefaulted = true - elsif cooling_system.cooling_system_type == HPXML::HVACTypeMiniSplitAirConditioner + when HPXML::HVACTypeMiniSplitAirConditioner cooling_system.cooling_shr = 0.73 cooling_system.cooling_shr_isdefaulted = true end @@ -2029,23 +2025,24 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu hpxml_bldg.heat_pumps.each do |heat_pump| next unless heat_pump.cooling_shr.nil? - if heat_pump.heat_pump_type == HPXML::HVACTypeHeatPumpAirToAir - if heat_pump.compressor_type == HPXML::HVACCompressorTypeSingleStage + case heat_pump.heat_pump_type + when HPXML::HVACTypeHeatPumpAirToAir + case heat_pump.compressor_type + when HPXML::HVACCompressorTypeSingleStage heat_pump.cooling_shr = 0.73 - elsif heat_pump.compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage heat_pump.cooling_shr = 0.73 - elsif heat_pump.compressor_type == HPXML::HVACCompressorTypeVariableSpeed + when HPXML::HVACCompressorTypeVariableSpeed heat_pump.cooling_shr = 0.78 end heat_pump.cooling_shr_isdefaulted = true - elsif heat_pump.heat_pump_type == HPXML::HVACTypeHeatPumpMiniSplit + when HPXML::HVACTypeHeatPumpMiniSplit heat_pump.cooling_shr = 0.73 heat_pump.cooling_shr_isdefaulted = true - elsif heat_pump.heat_pump_type == HPXML::HVACTypeHeatPumpGroundToAir + when HPXML::HVACTypeHeatPumpGroundToAir heat_pump.cooling_shr = 0.73 heat_pump.cooling_shr_isdefaulted = true - elsif heat_pump.heat_pump_type == HPXML::HVACTypeHeatPumpPTHP || - heat_pump.heat_pump_type == HPXML::HVACTypeHeatPumpRoom + when HPXML::HVACTypeHeatPumpPTHP, HPXML::HVACTypeHeatPumpRoom heat_pump.cooling_shr = 0.65 heat_pump.cooling_shr_isdefaulted = true end @@ -2111,7 +2108,8 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu mini_split_ductless_watts_per_cfm = 0.07 # W/cfm mini_split_ducted_watts_per_cfm = 0.18 # W/cfm hpxml_bldg.heating_systems.each do |heating_system| - if [HPXML::HVACTypeFurnace].include? heating_system.heating_system_type + case heating_system.heating_system_type + when HPXML::HVACTypeFurnace if heating_system.fan_watts_per_cfm.nil? if (not heating_system.distribution_system.nil?) && (heating_system.distribution_system.air_type == HPXML::AirTypeGravity) heating_system.fan_watts_per_cfm = 0.0 @@ -2122,15 +2120,13 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu end heating_system.fan_watts_per_cfm_isdefaulted = true end - elsif [HPXML::HVACTypeStove].include? heating_system.heating_system_type + when HPXML::HVACTypeStove if heating_system.fan_watts.nil? heating_system.fan_watts = 40.0 # W heating_system.fan_watts_isdefaulted = true end - elsif [HPXML::HVACTypeWallFurnace, - HPXML::HVACTypeFloorFurnace, - HPXML::HVACTypeSpaceHeater, - HPXML::HVACTypeFireplace].include? heating_system.heating_system_type + when HPXML::HVACTypeWallFurnace, HPXML::HVACTypeFloorFurnace, + HPXML::HVACTypeSpaceHeater, HPXML::HVACTypeFireplace if heating_system.fan_watts.nil? heating_system.fan_watts = 0.0 # W/cfm, assume no fan power heating_system.fan_watts_isdefaulted = true @@ -2143,42 +2139,46 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu if (not cooling_system.attached_heating_system.nil?) && (not cooling_system.attached_heating_system.fan_watts_per_cfm.nil?) cooling_system.fan_watts_per_cfm = cooling_system.attached_heating_system.fan_watts_per_cfm cooling_system.fan_watts_per_cfm_isdefaulted = true - elsif [HPXML::HVACTypeCentralAirConditioner].include? cooling_system.cooling_system_type - if cooling_system.cooling_efficiency_seer > 13.5 # HEScore assumption - cooling_system.fan_watts_per_cfm = ecm_watts_per_cfm - else - cooling_system.fan_watts_per_cfm = psc_watts_per_cfm - end - cooling_system.fan_watts_per_cfm_isdefaulted = true - elsif [HPXML::HVACTypeMiniSplitAirConditioner].include? cooling_system.cooling_system_type - if not cooling_system.distribution_system.nil? - cooling_system.fan_watts_per_cfm = mini_split_ducted_watts_per_cfm - else - cooling_system.fan_watts_per_cfm = mini_split_ductless_watts_per_cfm + else + case cooling_system.cooling_system_type + when HPXML::HVACTypeCentralAirConditioner + if cooling_system.cooling_efficiency_seer > 13.5 # HEScore assumption + cooling_system.fan_watts_per_cfm = ecm_watts_per_cfm + else + cooling_system.fan_watts_per_cfm = psc_watts_per_cfm + end + cooling_system.fan_watts_per_cfm_isdefaulted = true + when HPXML::HVACTypeMiniSplitAirConditioner + if not cooling_system.distribution_system.nil? + cooling_system.fan_watts_per_cfm = mini_split_ducted_watts_per_cfm + else + cooling_system.fan_watts_per_cfm = mini_split_ductless_watts_per_cfm + end + cooling_system.fan_watts_per_cfm_isdefaulted = true + when HPXML::HVACTypeEvaporativeCooler + # Depends on airflow rate, so defaulted in hvac_sizing.rb end - cooling_system.fan_watts_per_cfm_isdefaulted = true - elsif [HPXML::HVACTypeEvaporativeCooler].include? cooling_system.cooling_system_type - # Depends on airflow rate, so defaulted in hvac_sizing.rb end end hpxml_bldg.heat_pumps.each do |heat_pump| next unless heat_pump.fan_watts_per_cfm.nil? - if [HPXML::HVACTypeHeatPumpAirToAir].include? heat_pump.heat_pump_type + case heat_pump.heat_pump_type + when HPXML::HVACTypeHeatPumpAirToAir if heat_pump.heating_efficiency_hspf > 8.75 # HEScore assumption heat_pump.fan_watts_per_cfm = ecm_watts_per_cfm else heat_pump.fan_watts_per_cfm = psc_watts_per_cfm end heat_pump.fan_watts_per_cfm_isdefaulted = true - elsif [HPXML::HVACTypeHeatPumpGroundToAir].include? heat_pump.heat_pump_type + when HPXML::HVACTypeHeatPumpGroundToAir if heat_pump.heating_efficiency_cop > 8.75 / 3.2 # HEScore assumption heat_pump.fan_watts_per_cfm = ecm_watts_per_cfm else heat_pump.fan_watts_per_cfm = psc_watts_per_cfm end heat_pump.fan_watts_per_cfm_isdefaulted = true - elsif [HPXML::HVACTypeHeatPumpMiniSplit].include? heat_pump.heat_pump_type + when HPXML::HVACTypeHeatPumpMiniSplit if not heat_pump.distribution_system.nil? heat_pump.fan_watts_per_cfm = mini_split_ducted_watts_per_cfm else @@ -2234,10 +2234,9 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu # Detailed HVAC performance hpxml_bldg.cooling_systems.each do |cooling_system| clg_ap = cooling_system.additional_properties - if [HPXML::HVACTypeCentralAirConditioner, - HPXML::HVACTypeMiniSplitAirConditioner, - HPXML::HVACTypeRoomAirConditioner, - HPXML::HVACTypePTAC].include? cooling_system.cooling_system_type + case cooling_system.cooling_system_type + when HPXML::HVACTypeCentralAirConditioner, HPXML::HVACTypeMiniSplitAirConditioner, + HPXML::HVACTypeRoomAirConditioner, HPXML::HVACTypePTAC if [HPXML::HVACTypeRoomAirConditioner, HPXML::HVACTypePTAC].include? cooling_system.cooling_system_type use_eer = true @@ -2248,7 +2247,7 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu HVAC.set_fan_power_rated(cooling_system, use_eer) HVAC.set_cool_curves_central_air_source(cooling_system, use_eer) - elsif [HPXML::HVACTypeEvaporativeCooler].include? cooling_system.cooling_system_type + when HPXML::HVACTypeEvaporativeCooler clg_ap.effectiveness = 0.72 # Assumption from HEScore end @@ -2263,10 +2262,9 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu heating_system.additional_properties.heat_rated_cfm_per_ton = HVAC.get_heat_cfm_per_ton(HPXML::HVACCompressorTypeSingleStage, true) end hpxml_bldg.heat_pumps.each do |heat_pump| - if [HPXML::HVACTypeHeatPumpAirToAir, - HPXML::HVACTypeHeatPumpMiniSplit, - HPXML::HVACTypeHeatPumpPTHP, - HPXML::HVACTypeHeatPumpRoom].include? heat_pump.heat_pump_type + case heat_pump.heat_pump_type + when HPXML::HVACTypeHeatPumpAirToAir, HPXML::HVACTypeHeatPumpMiniSplit, + HPXML::HVACTypeHeatPumpPTHP, HPXML::HVACTypeHeatPumpRoom if [HPXML::HVACTypeHeatPumpPTHP, HPXML::HVACTypeHeatPumpRoom].include? heat_pump.heat_pump_type use_eer_cop = true else @@ -2277,7 +2275,7 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu HVAC.set_cool_curves_central_air_source(heat_pump, use_eer_cop) HVAC.set_heat_curves_central_air_source(heat_pump, use_eer_cop) - elsif [HPXML::HVACTypeHeatPumpGroundToAir].include? heat_pump.heat_pump_type + when HPXML::HVACTypeHeatPumpGroundToAir HVAC.set_heat_pump_temperatures(heat_pump, runner) if heat_pump.geothermal_loop.nil? @@ -2342,7 +2340,7 @@ def self.apply_hvac(runner, hpxml_bldg, weather, convert_shared_systems, unit_nu heat_pump.geothermal_loop.shank_spacing = (hp_ap.u_tube_spacing + hp_ap.pipe_od).round(2) # Distance from center of pipe to center of pipe heat_pump.geothermal_loop.shank_spacing_isdefaulted = true end - elsif [HPXML::HVACTypeHeatPumpWaterLoopToAir].include? heat_pump.heat_pump_type + when HPXML::HVACTypeHeatPumpWaterLoopToAir HVAC.set_heat_pump_temperatures(heat_pump, runner) end @@ -2993,12 +2991,13 @@ def self.apply_hot_water_distribution(hpxml_bldg, schedules_file) hot_water_distribution.pipe_r_value_isdefaulted = true end - if hot_water_distribution.system_type == HPXML::DHWDistTypeStandard + case hot_water_distribution.system_type + when HPXML::DHWDistTypeStandard if hot_water_distribution.standard_piping_length.nil? hot_water_distribution.standard_piping_length = get_std_pipe_length(has_uncond_bsmnt, has_cond_bsmnt, cfa, ncfl) hot_water_distribution.standard_piping_length_isdefaulted = true end - elsif hot_water_distribution.system_type == HPXML::DHWDistTypeRecirc + when HPXML::DHWDistTypeRecirc if hot_water_distribution.recirculation_piping_loop_length.nil? hot_water_distribution.recirculation_piping_loop_length = get_recirc_loop_length(has_uncond_bsmnt, has_cond_bsmnt, cfa, ncfl) hot_water_distribution.recirculation_piping_loop_length_isdefaulted = true @@ -3023,7 +3022,8 @@ def self.apply_hot_water_distribution(hpxml_bldg, schedules_file) if hot_water_distribution.system_type == HPXML::DHWDistTypeRecirc || hot_water_distribution.has_shared_recirculation schedules_file_includes_recirculation_pump = (schedules_file.nil? ? false : schedules_file.includes_col_name(SchedulesFile::Columns[:HotWaterRecirculationPump].name)) recirc_control_type = hot_water_distribution.has_shared_recirculation ? hot_water_distribution.shared_recirculation_control_type : hot_water_distribution.recirculation_control_type - if [HPXML::DHWRecircControlTypeNone, HPXML::DHWRecircControlTypeTimer].include?(recirc_control_type) + case recirc_control_type + when HPXML::DHWRecircControlTypeNone, HPXML::DHWRecircControlTypeTimer if hot_water_distribution.recirculation_pump_weekday_fractions.nil? && !schedules_file_includes_recirculation_pump hot_water_distribution.recirculation_pump_weekday_fractions = @default_schedules_csv_data["#{SchedulesFile::Columns[:HotWaterRecirculationPump].name}_no_control"]['RecirculationPumpWeekdayScheduleFractions'] hot_water_distribution.recirculation_pump_weekday_fractions_isdefaulted = true @@ -3032,7 +3032,7 @@ def self.apply_hot_water_distribution(hpxml_bldg, schedules_file) hot_water_distribution.recirculation_pump_weekend_fractions = @default_schedules_csv_data["#{SchedulesFile::Columns[:HotWaterRecirculationPump].name}_no_control"]['RecirculationPumpWeekendScheduleFractions'] hot_water_distribution.recirculation_pump_weekend_fractions_isdefaulted = true end - elsif [HPXML::DHWRecircControlTypeSensor, HPXML::DHWRecircControlTypeManual].include?(recirc_control_type) + when HPXML::DHWRecircControlTypeSensor, HPXML::DHWRecircControlTypeManual if hot_water_distribution.recirculation_pump_weekday_fractions.nil? && !schedules_file_includes_recirculation_pump hot_water_distribution.recirculation_pump_weekday_fractions = @default_schedules_csv_data["#{SchedulesFile::Columns[:HotWaterRecirculationPump].name}_demand_control"]['RecirculationPumpWeekdayScheduleFractions'] hot_water_distribution.recirculation_pump_weekday_fractions_isdefaulted = true @@ -3041,7 +3041,7 @@ def self.apply_hot_water_distribution(hpxml_bldg, schedules_file) hot_water_distribution.recirculation_pump_weekend_fractions = @default_schedules_csv_data["#{SchedulesFile::Columns[:HotWaterRecirculationPump].name}_demand_control"]['RecirculationPumpWeekendScheduleFractions'] hot_water_distribution.recirculation_pump_weekend_fractions_isdefaulted = true end - elsif [HPXML::DHWRecircControlTypeTemperature].include?(recirc_control_type) + when HPXML::DHWRecircControlTypeTemperature if hot_water_distribution.recirculation_pump_weekday_fractions.nil? && !schedules_file_includes_recirculation_pump hot_water_distribution.recirculation_pump_weekday_fractions = @default_schedules_csv_data["#{SchedulesFile::Columns[:HotWaterRecirculationPump].name}_temperature_control"]['RecirculationPumpWeekdayScheduleFractions'] hot_water_distribution.recirculation_pump_weekday_fractions_isdefaulted = true @@ -3682,7 +3682,9 @@ def self.apply_ceiling_fans(hpxml_bldg, weather, schedules_file) # @param schedules_file [SchedulesFile] SchedulesFile wrapper class instance of detailed schedule files # @return [nil] def self.apply_pools_and_permanent_spas(hpxml_bldg, schedules_file) - nbeds_eq = hpxml_bldg.building_construction.additional_properties.equivalent_number_of_bedrooms + nbeds = hpxml_bldg.building_construction.number_of_bedrooms + n_occ = hpxml_bldg.building_occupancy.number_of_residents + unit_type = hpxml_bldg.building_construction.residential_facility_type cfa = hpxml_bldg.building_construction.conditioned_floor_area hpxml_bldg.pools.each do |pool| next if pool.type == HPXML::TypeNone @@ -3690,7 +3692,7 @@ def self.apply_pools_and_permanent_spas(hpxml_bldg, schedules_file) if pool.pump_type != HPXML::TypeNone # Pump if pool.pump_kwh_per_year.nil? - pool.pump_kwh_per_year = get_pool_pump_annual_energy(cfa, nbeds_eq) + pool.pump_kwh_per_year = get_pool_pump_annual_energy(cfa, nbeds, n_occ, unit_type) pool.pump_kwh_per_year_isdefaulted = true end if pool.pump_usage_multiplier.nil? @@ -3716,7 +3718,7 @@ def self.apply_pools_and_permanent_spas(hpxml_bldg, schedules_file) # Heater if pool.heater_load_value.nil? - default_heater_load_units, default_heater_load_value = get_pool_heater_annual_energy(cfa, nbeds_eq, pool.heater_type) + default_heater_load_units, default_heater_load_value = get_pool_heater_annual_energy(cfa, nbeds, n_occ, unit_type, pool.heater_type) pool.heater_load_units = default_heater_load_units pool.heater_load_value = default_heater_load_value pool.heater_load_value_isdefaulted = true @@ -3746,7 +3748,7 @@ def self.apply_pools_and_permanent_spas(hpxml_bldg, schedules_file) if spa.pump_type != HPXML::TypeNone # Pump if spa.pump_kwh_per_year.nil? - spa.pump_kwh_per_year = get_permanent_spa_pump_annual_energy(cfa, nbeds_eq) + spa.pump_kwh_per_year = get_permanent_spa_pump_annual_energy(cfa, nbeds, n_occ, unit_type) spa.pump_kwh_per_year_isdefaulted = true end if spa.pump_usage_multiplier.nil? @@ -3772,7 +3774,7 @@ def self.apply_pools_and_permanent_spas(hpxml_bldg, schedules_file) # Heater if spa.heater_load_value.nil? - default_heater_load_units, default_heater_load_value = get_permanent_spa_heater_annual_energy(cfa, nbeds_eq, spa.heater_type) + default_heater_load_units, default_heater_load_value = get_permanent_spa_heater_annual_energy(cfa, nbeds, n_occ, unit_type, spa.heater_type) spa.heater_load_units = default_heater_load_units spa.heater_load_value = default_heater_load_value spa.heater_load_value_isdefaulted = true @@ -3805,12 +3807,12 @@ def self.apply_pools_and_permanent_spas(hpxml_bldg, schedules_file) def self.apply_plug_loads(hpxml_bldg, schedules_file) cfa = hpxml_bldg.building_construction.conditioned_floor_area nbeds = hpxml_bldg.building_construction.number_of_bedrooms - nbeds_eq = hpxml_bldg.building_construction.additional_properties.equivalent_number_of_bedrooms - num_occ = hpxml_bldg.building_occupancy.number_of_residents + n_occ = hpxml_bldg.building_occupancy.number_of_residents unit_type = hpxml_bldg.building_construction.residential_facility_type hpxml_bldg.plug_loads.each do |plug_load| - if plug_load.plug_load_type == HPXML::PlugLoadTypeOther - default_annual_kwh, default_sens_frac, default_lat_frac = get_residual_mels_values(cfa, num_occ, unit_type) + case plug_load.plug_load_type + when HPXML::PlugLoadTypeOther + default_annual_kwh, default_sens_frac, default_lat_frac = get_residual_mels_values(cfa, n_occ, unit_type) if plug_load.kwh_per_year.nil? plug_load.kwh_per_year = default_annual_kwh plug_load.kwh_per_year_isdefaulted = true @@ -3836,8 +3838,8 @@ def self.apply_plug_loads(hpxml_bldg, schedules_file) plug_load.monthly_multipliers = @default_schedules_csv_data[SchedulesFile::Columns[:PlugLoadsOther].name]['MonthlyScheduleMultipliers'] plug_load.monthly_multipliers_isdefaulted = true end - elsif plug_load.plug_load_type == HPXML::PlugLoadTypeTelevision - default_annual_kwh, default_sens_frac, default_lat_frac = get_televisions_values(cfa, nbeds, num_occ, unit_type) + when HPXML::PlugLoadTypeTelevision + default_annual_kwh, default_sens_frac, default_lat_frac = get_televisions_values(cfa, nbeds, n_occ, unit_type) if plug_load.kwh_per_year.nil? plug_load.kwh_per_year = default_annual_kwh plug_load.kwh_per_year_isdefaulted = true @@ -3863,7 +3865,7 @@ def self.apply_plug_loads(hpxml_bldg, schedules_file) plug_load.monthly_multipliers = @default_schedules_csv_data[SchedulesFile::Columns[:PlugLoadsTV].name]['MonthlyScheduleMultipliers'] plug_load.monthly_multipliers_isdefaulted = true end - elsif plug_load.plug_load_type == HPXML::PlugLoadTypeElectricVehicleCharging + when HPXML::PlugLoadTypeElectricVehicleCharging default_annual_kwh = get_electric_vehicle_charging_annual_energy if plug_load.kwh_per_year.nil? plug_load.kwh_per_year = default_annual_kwh @@ -3890,8 +3892,8 @@ def self.apply_plug_loads(hpxml_bldg, schedules_file) plug_load.monthly_multipliers = @default_schedules_csv_data[SchedulesFile::Columns[:PlugLoadsVehicle].name]['MonthlyScheduleMultipliers'] plug_load.monthly_multipliers_isdefaulted = true end - elsif plug_load.plug_load_type == HPXML::PlugLoadTypeWellPump - default_annual_kwh = get_detault_well_pump_annual_energy(cfa, nbeds_eq) + when HPXML::PlugLoadTypeWellPump + default_annual_kwh = get_detault_well_pump_annual_energy(cfa, nbeds, n_occ, unit_type) if plug_load.kwh_per_year.nil? plug_load.kwh_per_year = default_annual_kwh plug_load.kwh_per_year_isdefaulted = true @@ -3932,11 +3934,14 @@ def self.apply_plug_loads(hpxml_bldg, schedules_file) # @return [nil] def self.apply_fuel_loads(hpxml_bldg, schedules_file) cfa = hpxml_bldg.building_construction.conditioned_floor_area - nbeds_eq = hpxml_bldg.building_construction.additional_properties.equivalent_number_of_bedrooms + nbeds = hpxml_bldg.building_construction.number_of_bedrooms + n_occ = hpxml_bldg.building_occupancy.number_of_residents + unit_type = hpxml_bldg.building_construction.residential_facility_type hpxml_bldg.fuel_loads.each do |fuel_load| - if fuel_load.fuel_load_type == HPXML::FuelLoadTypeGrill + case fuel_load.fuel_load_type + when HPXML::FuelLoadTypeGrill if fuel_load.therm_per_year.nil? - fuel_load.therm_per_year = get_gas_grill_annual_energy(cfa, nbeds_eq) + fuel_load.therm_per_year = get_gas_grill_annual_energy(cfa, nbeds, n_occ, unit_type) fuel_load.therm_per_year_isdefaulted = true end if fuel_load.frac_sensible.nil? @@ -3960,9 +3965,9 @@ def self.apply_fuel_loads(hpxml_bldg, schedules_file) fuel_load.monthly_multipliers = @default_schedules_csv_data[SchedulesFile::Columns[:FuelLoadsGrill].name]['MonthlyScheduleMultipliers'] fuel_load.monthly_multipliers_isdefaulted = true end - elsif fuel_load.fuel_load_type == HPXML::FuelLoadTypeLighting + when HPXML::FuelLoadTypeLighting if fuel_load.therm_per_year.nil? - fuel_load.therm_per_year = get_detault_gas_lighting_annual_energy(cfa, nbeds_eq) + fuel_load.therm_per_year = get_detault_gas_lighting_annual_energy(cfa, nbeds, n_occ, unit_type) fuel_load.therm_per_year_isdefaulted = true end if fuel_load.frac_sensible.nil? @@ -3986,9 +3991,9 @@ def self.apply_fuel_loads(hpxml_bldg, schedules_file) fuel_load.monthly_multipliers = @default_schedules_csv_data[SchedulesFile::Columns[:FuelLoadsLighting].name]['MonthlyScheduleMultipliers'] fuel_load.monthly_multipliers_isdefaulted = true end - elsif fuel_load.fuel_load_type == HPXML::FuelLoadTypeFireplace + when HPXML::FuelLoadTypeFireplace if fuel_load.therm_per_year.nil? - fuel_load.therm_per_year = get_gas_fireplace_annual_energy(cfa, nbeds_eq) + fuel_load.therm_per_year = get_gas_fireplace_annual_energy(cfa, nbeds, n_occ, unit_type) fuel_load.therm_per_year_isdefaulted = true end if fuel_load.frac_sensible.nil? @@ -4050,21 +4055,22 @@ def self.cleanup_zones_spaces(hpxml_bldg) def self.get_azimuth_from_orientation(orientation) return if orientation.nil? - if orientation == HPXML::OrientationNorth + case orientation + when HPXML::OrientationNorth return 0 - elsif orientation == HPXML::OrientationNortheast + when HPXML::OrientationNortheast return 45 - elsif orientation == HPXML::OrientationEast + when HPXML::OrientationEast return 90 - elsif orientation == HPXML::OrientationSoutheast + when HPXML::OrientationSoutheast return 135 - elsif orientation == HPXML::OrientationSouth + when HPXML::OrientationSouth return 180 - elsif orientation == HPXML::OrientationSouthwest + when HPXML::OrientationSouthwest return 225 - elsif orientation == HPXML::OrientationWest + when HPXML::OrientationWest return 270 - elsif orientation == HPXML::OrientationNorthwest + when HPXML::OrientationNorthwest return 315 end @@ -4098,26 +4104,34 @@ def self.get_orientation_from_azimuth(azimuth) end # Gets the equivalent number of bedrooms for an operational calculation (i.e., when number - # of occupants are provided in the HPXML); this is an adjustment to the ANSI 301 or Building - # America equations, which are based on number of bedrooms. + # of occupants are provided in the HPXML); this is an adjustment to the ANSI/RESNET/ICC 301 or Building + # America equations, which are based on number of bedrooms. If an asset calculation (number + # of occupants provided), the number of bedrooms is simply returned. # # This is used to adjust occupancy-driven end uses from asset calculations (based on number # of bedrooms) to operational calculations (based on number of occupants). # - # @param hpxml_bldg [HPXML::Building] HPXML Building object representing an individual dwelling unit + # Source: 2020 RECS weighted regressions between NBEDS and NHSHLDMEM (sample weights = NWEIGHT) + # + # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @return [Double] Equivalent number of bedrooms - def self.get_equivalent_nbeds_for_operational_calculation(hpxml_bldg) - n_occs = hpxml_bldg.building_occupancy.number_of_residents - unit_type = hpxml_bldg.building_construction.residential_facility_type - # Relations below come from 2020 RECS weighted regressions between NBEDS and NHSHLDMEM (sample weights = NWEIGHT) - if [HPXML::ResidentialTypeApartment].include? unit_type - return -1.36 + 1.49 * n_occs - elsif [HPXML::ResidentialTypeSFA].include? unit_type - return -1.98 + 1.89 * n_occs - elsif [HPXML::ResidentialTypeSFD].include? unit_type - return -2.19 + 2.08 * n_occs - elsif [HPXML::ResidentialTypeManufactured].include? unit_type - return -1.26 + 1.61 * n_occs + def self.get_equivalent_nbeds(nbeds, n_occ, unit_type) + if n_occ.nil? + # No occupants specified, asset rating + return nbeds + end + + case unit_type + when HPXML::ResidentialTypeApartment + return -1.36 + 1.49 * n_occ + when HPXML::ResidentialTypeSFA + return -1.98 + 1.89 * n_occ + when HPXML::ResidentialTypeSFD + return -2.19 + 2.08 * n_occ + when HPXML::ResidentialTypeManufactured + return -1.26 + 1.61 * n_occ else fail "Unexpected residential facility type: #{unit_type}." end @@ -4134,20 +4148,16 @@ def self.get_flue_or_chimney_in_conditioned_space(hpxml_bldg) next if heating_system.heating_system_fuel == HPXML::FuelTypeElectricity next unless HPXML::conditioned_locations_this_unit.include? heating_system.location - if [HPXML::HVACTypeFurnace, - HPXML::HVACTypeBoiler, - HPXML::HVACTypeWallFurnace, - HPXML::HVACTypeFloorFurnace, - HPXML::HVACTypeStove, - HPXML::HVACTypeSpaceHeater].include? heating_system.heating_system_type + case heating_system.heating_system_type + when HPXML::HVACTypeFurnace, HPXML::HVACTypeBoiler, HPXML::HVACTypeWallFurnace, + HPXML::HVACTypeFloorFurnace, HPXML::HVACTypeStove, HPXML::HVACTypeSpaceHeater if not heating_system.heating_efficiency_afue.nil? next if heating_system.heating_efficiency_afue >= 0.89 elsif not heating_system.heating_efficiency_percent.nil? next if heating_system.heating_efficiency_percent >= 0.89 end - return true - elsif [HPXML::HVACTypeFireplace].include? heating_system.heating_system_type + when HPXML::HVACTypeFireplace return true end end @@ -4575,11 +4585,13 @@ def self.get_clothes_washer_values(eri_version) # Gets the default piping length for a standard hot water distribution system. # - # Per ANSI 301-2022, the length of hot water piping from the hot water heater to the farthest + # The length of hot water piping from the hot water heater to the farthest # hot water fixture, measured longitudinally from plans, assuming the hot water piping does # not run diagonally, plus 10 feet of piping for each floor level, plus 5 feet of piping for # unconditioned basements (if any). # + # Source: ANSI/RESNET/ICC 301-2022 + # # @param has_uncond_bsmnt [Boolean] Whether the dwelling unit has an unconditioned basement # @param has_cond_bsmnt [Boolean] Whether the dwelling unit has a conditioned basement # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) @@ -4591,16 +4603,18 @@ def self.get_std_pipe_length(has_uncond_bsmnt, has_cond_bsmnt, cfa, ncfl) bsmnt = 1 end - return 2.0 * (cfa / ncfl)**0.5 + 10.0 * ncfl + 5.0 * bsmnt # PipeL in ANSI 301 + return 2.0 * (cfa / ncfl)**0.5 + 10.0 * ncfl + 5.0 * bsmnt # PipeL in ANSI/RESNET/ICC 301 end # Gets the default loop piping length for a recirculation hot water distribution system. # - # Per ANSI 301-2022, the recirculation loop length including both supply and return sides, + # The recirculation loop length including both supply and return sides, # measured longitudinally from plans, assuming the hot water piping does not run diagonally, # plus 20 feet of piping for each floor level greater than one plus 10 feet of piping for # unconditioned basements. # + # Source: ANSI/RESNET/ICC 301-2022 + # # @param has_uncond_bsmnt [Boolean] Whether the dwelling unit has an unconditioned basement # @param has_cond_bsmnt [Boolean] Whether the dwelling unit has a conditioned basement # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) @@ -4608,32 +4622,34 @@ def self.get_std_pipe_length(has_uncond_bsmnt, has_cond_bsmnt, cfa, ncfl) # @return [Double] Piping length (ft) def self.get_recirc_loop_length(has_uncond_bsmnt, has_cond_bsmnt, cfa, ncfl) std_pipe_length = get_std_pipe_length(has_uncond_bsmnt, has_cond_bsmnt, cfa, ncfl) - return 2.0 * std_pipe_length - 20.0 # refLoopL in ANSI 301 + return 2.0 * std_pipe_length - 20.0 # refLoopL in ANSI/RESNET/ICC 301 end # Gets the default branch piping length for a recirculation hot water distribution system. # - # Per ANSI 301-2022, the length of the branch hot water piping from the recirculation loop + # The length of the branch hot water piping from the recirculation loop # to the farthest hot water fixture from the recirculation loop, measured longitudinally # from plans, assuming the branch hot water piping does not run diagonally. # + # Source: ANSI/RESNET/ICC 301-2022 + # # @return [Double] Piping length (ft) def self.get_recirc_branch_length() - return 10.0 # See pRatio in ANSI 301 + return 10.0 # See pRatio in ANSI/RESNET/ICC 301 end # Gets the default pump power for a recirculation system. # # @return [Double] Pump power (W) def self.get_recirc_pump_power() - return 50.0 # See pumpW in ANSI 301 + return 50.0 # See pumpW in ANSI/RESNET/ICC 301 end # Gets the default pump power for a shared recirculation system. # # @return [Double] Pump power (W) def self.get_shared_recirc_pump_power() - # From ANSI/RESNET/ICC 301-2019 Equation 4.2-15b + # From ANSI/RESNET/ICC 301-2022 Eq. 4.2-43b pump_horsepower = 0.25 motor_efficiency = 0.85 pump_kw = pump_horsepower * 0.746 / motor_efficiency @@ -4667,9 +4683,11 @@ def self.get_freezer_or_extra_fridge_location(hpxml_bldg) # Window represents multiple windows, the value is calculated as the total window area # for any operable windows divided by the total window area. # + # Source: ANSI/RESNET/ICC 301-2025 + # # @return [Double] Operable fraction (frac) def self.get_fraction_of_windows_operable() - return 0.67 # 67% per ANSI 301-2025 + return 0.67 # 67% end # Gets the default specific leakage area (SLA) for a vented attic. @@ -4677,7 +4695,7 @@ def self.get_fraction_of_windows_operable() # # @return [Double] Specific leakage area (frac) def self.get_vented_attic_sla() - return (1.0 / 300.0).round(6) # ANSI 301, Table 4.2.2(1) - Attics + return (1.0 / 300.0).round(6) # ANSI/RESNET/ICC 301, Table 4.2.2(1) - Attics end # Gets the default specific leakage area (SLA) for a vented crawlspace. @@ -4685,7 +4703,7 @@ def self.get_vented_attic_sla() # # @return [Double] Specific leakage area (frac) def self.get_vented_crawl_sla() - return (1.0 / 150.0).round(6) # ANSI 301, Table 4.2.2(1) - Crawlspaces + return (1.0 / 150.0).round(6) # ANSI/RESNET/ICC 301, Table 4.2.2(1) - Crawlspaces end # Gets the default whole-home mechanical ventilation fan flow rate required to @@ -4700,11 +4718,11 @@ def self.get_vented_crawl_sla() # @param eri_version [String] Version of the ANSI/RESNET/ICC 301 Standard to use for equations/assumptions # @return [Double] Fan flow rate (cfm) def self.get_mech_vent_flow_rate_for_vent_fan(hpxml_bldg, vent_fan, weather, eri_version) - # Calculates Qfan cfm requirement per ASHRAE 62.2 / ANSI 301 + # Calculates Qfan cfm requirement per ASHRAE 62.2 / ANSI/RESNET/ICC 301 cfa = hpxml_bldg.building_construction.conditioned_floor_area nbeds = hpxml_bldg.building_construction.number_of_bedrooms infil_values = Airflow.get_values_from_air_infiltration_measurements(hpxml_bldg, weather) - bldg_type = hpxml_bldg.building_construction.residential_facility_type + unit_type = hpxml_bldg.building_construction.residential_facility_type nl = Airflow.get_infiltration_NL_from_SLA(infil_values[:sla], infil_values[:height]) q_inf = Airflow.get_infiltration_Qinf_from_NL(nl, weather, cfa) @@ -4714,23 +4732,27 @@ def self.get_mech_vent_flow_rate_for_vent_fan(hpxml_bldg, vent_fan, weather, eri else is_balanced, frac_imbal = false, 1.0 end - q_fan = Airflow.get_mech_vent_qfan_cfm(q_tot, q_inf, is_balanced, frac_imbal, infil_values[:a_ext], bldg_type, eri_version, vent_fan.hours_in_operation) + q_fan = Airflow.get_mech_vent_qfan_cfm(q_tot, q_inf, is_balanced, frac_imbal, infil_values[:a_ext], unit_type, eri_version, vent_fan.hours_in_operation) return q_fan end # Gets the default whole-home mechanical ventilation fan efficiency. # + # Source: ANSI/RESNET/ICC 301 + # # @param vent_fan [HPXML::VentilationFan] The HPXML ventilation fan of interest # @return [Double] Fan efficiency (W/cfm) def self.get_mech_vent_fan_efficiency(vent_fan) - # Returns fan power in W/cfm, based on ANSI 301 if vent_fan.is_shared_system return 1.00 # Table 4.2.2(1) Note (n) - elsif [HPXML::MechVentTypeSupply, HPXML::MechVentTypeExhaust].include? vent_fan.fan_type + end + + case vent_fan.fan_type + when HPXML::MechVentTypeSupply, HPXML::MechVentTypeExhaust return 0.35 - elsif [HPXML::MechVentTypeBalanced].include? vent_fan.fan_type + when HPXML::MechVentTypeBalanced return 0.70 - elsif [HPXML::MechVentTypeERV, HPXML::MechVentTypeHRV].include? vent_fan.fan_type + when HPXML::MechVentTypeERV, HPXML::MechVentTypeHRV return 1.00 else fail "Unexpected fan_type: '#{fan_type}'." @@ -4788,27 +4810,28 @@ def self.get_infiltration_ach50(cfa, ncfl_ag, year_built, avg_ceiling_height, in # Climate zone c_iecc = nil - if (iecc_cz == '1A') || (iecc_cz == '2A') + case iecc_cz + when '1A', '2A' c_iecc = 0.4727 - elsif iecc_cz == '3A' + when '3A' c_iecc = 0.2529 - elsif iecc_cz == '4A' + when '4A' c_iecc = 0.3261 - elsif iecc_cz == '5A' + when '5A' c_iecc = 0.1118 - elsif (iecc_cz == '6A') || (iecc_cz == '7') + when '6A', '7' c_iecc = 0.0 - elsif (iecc_cz == '2B') || (iecc_cz == '3B') + when '2B', '3B' c_iecc = -0.03755 - elsif (iecc_cz == '4B') || (iecc_cz == '5B') + when '4B', '5B' c_iecc = -0.008774 - elsif iecc_cz == '6B' + when '6B' c_iecc = 0.01944 - elsif iecc_cz == '3C' + when '3C' c_iecc = 0.04827 - elsif iecc_cz == '4C' + when '4C' c_iecc = 0.2584 - elsif iecc_cz == '8' + when '8' c_iecc = -0.5119 else fail "Unexpected IECC climate zone: #{c_iecc}" @@ -4869,7 +4892,7 @@ def self.get_infiltration_ach50(cfa, ncfl_ag, year_built, avg_ceiling_height, in # @param f_rect [Double] The fraction of duct length that is rectangular (not round) # @return [Double] Duct effective R-value (hr-ft2-F/Btu) def self.get_duct_effective_r_value(r_nominal, side, buried_level, f_rect) - # This methodology has been proposed by NREL for ANSI 301-2025. + # This methodology has been proposed by NREL for ANSI/RESNET/ICC 301-2025. if buried_level == HPXML::DuctBuriedInsulationNone if r_nominal <= 0 # Uninsulated ducts are set to R-1.7 based on ASHRAE HOF and the above paper. @@ -4898,21 +4921,23 @@ def self.get_duct_effective_r_value(r_nominal, side, buried_level, f_rect) if side == HPXML::DuctTypeSupply # Equations derived from Table 13 in https://www.nrel.gov/docs/fy13osti/55876.pdf # assuming 6-in supply diameter - if buried_level == HPXML::DuctBuriedInsulationPartial + case buried_level + when HPXML::DuctBuriedInsulationPartial return (4.28 + 0.65 * r_nominal).round(2) - elsif buried_level == HPXML::DuctBuriedInsulationFull + when HPXML::DuctBuriedInsulationFull return (6.22 + 0.89 * r_nominal).round(2) - elsif buried_level == HPXML::DuctBuriedInsulationDeep + when HPXML::DuctBuriedInsulationDeep return (13.41 + 0.63 * r_nominal).round(2) end elsif side == HPXML::DuctTypeReturn # Equations derived from Table 13 in https://www.nrel.gov/docs/fy13osti/55876.pdf # assuming 14-in return diameter - if buried_level == HPXML::DuctBuriedInsulationPartial + case buried_level + when HPXML::DuctBuriedInsulationPartial return (4.62 + 1.31 * r_nominal).round(2) - elsif buried_level == HPXML::DuctBuriedInsulationFull + when HPXML::DuctBuriedInsulationFull return (8.91 + 1.29 * r_nominal).round(2) - elsif buried_level == HPXML::DuctBuriedInsulationDeep + when HPXML::DuctBuriedInsulationDeep return (18.64 + 1.0 * r_nominal).round(2) end end @@ -4926,14 +4951,19 @@ def self.get_duct_effective_r_value(r_nominal, side, buried_level, f_rect) # @return [String] Water heater location (HPXML::LocationXXX) def self.get_water_heater_location(hpxml_bldg, iecc_zone = nil) # ANSI/RESNET/ICC 301-2022C - if ['1A', '1B', '1C', '2A', '2B', '2C', '3A', '3B', '3C'].include? iecc_zone + case iecc_zone + when '1A', '1B', '1C', '2A', '2B', '2C', '3A', '3B', '3C' location_hierarchy = [HPXML::LocationGarage, HPXML::LocationConditionedSpace] - elsif ['4A', '4B', '4C', '5A', '5B', '5C', '6A', '6B', '6C', '7', '8'].include? iecc_zone + when '4A', '4B', '4C', '5A', '5B', '5C', '6A', '6B', '6C', '7', '8' location_hierarchy = [HPXML::LocationBasementUnconditioned, HPXML::LocationBasementConditioned, HPXML::LocationConditionedSpace] - elsif iecc_zone.nil? + else + if not iecc_zone.nil? + fail "Unexpected IECC zone: #{iecc_zone}." + end + location_hierarchy = [HPXML::LocationBasementConditioned, HPXML::LocationBasementUnconditioned, HPXML::LocationConditionedSpace] @@ -4966,9 +4996,9 @@ def self.get_water_heater_temperature(eri_version) def self.get_water_heater_performance_adjustment(water_heating_system) return unless water_heating_system.water_heater_type == HPXML::WaterHeaterTypeTankless if not water_heating_system.energy_factor.nil? - return 0.92 # Applies to EF, ANSI 301-2019 + return 0.92 # Applies to EF, ANSI/RESNET/ICC 301-2022 elsif not water_heating_system.uniform_energy_factor.nil? - return 0.94 # Applies to UEF, ANSI 301-2019 + return 0.94 # Applies to UEF, ANSI/RESNET/ICC 301-2022 end end @@ -5038,15 +5068,16 @@ def self.get_water_heater_tank_volume(fuel, nbeds, nbaths = nil) end if fuel != HPXML::FuelTypeElectricity # Non-electric tank WHs - if nbeds <= 2 + case nbeds + when 0, 1, 2 return 30.0 - elsif nbeds == 3 + when 3 if nbaths <= 1.5 return 30.0 else return 40.0 end - elsif nbeds == 4 + when 4 if nbaths <= 2.5 return 40.0 else @@ -5056,27 +5087,28 @@ def self.get_water_heater_tank_volume(fuel, nbeds, nbaths = nil) return 50.0 end else - if nbeds == 1 + case nbeds + when 0, 1 return 30.0 - elsif nbeds == 2 + when 2 if nbaths <= 1.5 return 30.0 else return 40.0 end - elsif nbeds == 3 + when 3 if nbaths <= 1.5 return 40.0 else return 50.0 end - elsif nbeds == 4 + when 4 if nbaths <= 2.5 return 50.0 else return 66.0 end - elsif nbeds == 5 + when 5 return 66.0 else return 80.0 @@ -5195,37 +5227,34 @@ def self.get_wall_solar_absorptance(wall) def self.get_window_ufactor_shgc(window) type = window.is_a?(HPXML::Window) ? 'window' : 'skylight' - if window.glass_layers == HPXML::WindowLayersSinglePane + case window.glass_layers + when HPXML::WindowLayersSinglePane n_panes = 1 - elsif window.glass_layers == HPXML::WindowLayersDoublePane + when HPXML::WindowLayersDoublePane n_panes = 2 - elsif window.glass_layers == HPXML::WindowLayersTriplePane + when HPXML::WindowLayersTriplePane n_panes = 3 - elsif window.glass_layers == HPXML::WindowLayersGlassBlock + when HPXML::WindowLayersGlassBlock return [0.6, 0.6] # From https://www.federalregister.gov/documents/2016/06/17/2016-13547/energy-conservation-standards-for-manufactured-housing end - if [HPXML::WindowFrameTypeAluminum, - HPXML::WindowFrameTypeMetal].include? window.frame_type + case window.frame_type + when HPXML::WindowFrameTypeAluminum, HPXML::WindowFrameTypeMetal is_metal_frame = true - elsif [HPXML::WindowFrameTypeWood, - HPXML::WindowFrameTypeVinyl, - HPXML::WindowFrameTypeFiberglass].include? window.frame_type + when HPXML::WindowFrameTypeWood, HPXML::WindowFrameTypeVinyl, HPXML::WindowFrameTypeFiberglass is_metal_frame = false else fail "Unexpected #{type.downcase} frame type." end - if [HPXML::WindowGlassTypeClear, - HPXML::WindowGlassTypeReflective].include? window.glass_type + case window.glass_type + when HPXML::WindowGlassTypeClear, HPXML::WindowGlassTypeReflective glass_type = 'clear' - elsif [HPXML::WindowGlassTypeTinted, - HPXML::WindowGlassTypeTintedReflective].include? window.glass_type + when HPXML::WindowGlassTypeTinted, HPXML::WindowGlassTypeTintedReflective glass_type = 'tinted' - elsif [HPXML::WindowGlassTypeLowE, - HPXML::WindowGlassTypeLowEHighSolarGain].include? window.glass_type + when HPXML::WindowGlassTypeLowE, HPXML::WindowGlassTypeLowEHighSolarGain glass_type = 'low_e_insulating' - elsif [HPXML::WindowGlassTypeLowELowSolarGain].include? window.glass_type + when HPXML::WindowGlassTypeLowELowSolarGain glass_type = 'low_e_solar_control' else fail "Unexpected #{type.downcase} glass type." @@ -5233,16 +5262,19 @@ def self.get_window_ufactor_shgc(window) if window.glass_layers == HPXML::WindowLayersSinglePane gas_fill = 'none' - elsif [HPXML::WindowGasAir].include? window.gas_fill - gas_fill = 'air' - elsif [HPXML::WindowGasArgon, + else + case window.gas_fill + when HPXML::WindowGasAir + gas_fill = 'air' + when HPXML::WindowGasArgon, HPXML::WindowGasKrypton, HPXML::WindowGasXenon, HPXML::WindowGasNitrogen, - HPXML::WindowGasOther].include? window.gas_fill - gas_fill = 'gas' - else - fail "Unexpected #{type.downcase} gas type." + HPXML::WindowGasOther + gas_fill = 'gas' + else + fail "Unexpected #{type.downcase} gas type." + end end # Lookup values @@ -5300,8 +5332,9 @@ def self.get_window_ufactor_shgc(window) # @param seer [Double] Cooling efficiency # @return [String] Compressor type (HPXML::HVACCompressorTypeXXX) def self.get_hvac_compressor_type(hvac_type, seer) - if [HPXML::HVACTypeCentralAirConditioner, - HPXML::HVACTypeHeatPumpAirToAir].include? hvac_type + case hvac_type + when HPXML::HVACTypeCentralAirConditioner, + HPXML::HVACTypeHeatPumpAirToAir if seer <= 15 return HPXML::HVACCompressorTypeSingleStage elsif seer <= 21 @@ -5309,13 +5342,13 @@ def self.get_hvac_compressor_type(hvac_type, seer) elsif seer > 21 return HPXML::HVACCompressorTypeVariableSpeed end - elsif [HPXML::HVACTypeMiniSplitAirConditioner, - HPXML::HVACTypeHeatPumpMiniSplit].include? hvac_type + when HPXML::HVACTypeMiniSplitAirConditioner, + HPXML::HVACTypeHeatPumpMiniSplit return HPXML::HVACCompressorTypeVariableSpeed - elsif [HPXML::HVACTypePTAC, - HPXML::HVACTypeHeatPumpPTHP, - HPXML::HVACTypeHeatPumpRoom, - HPXML::HVACTypeRoomAirConditioner].include? hvac_type + when HPXML::HVACTypePTAC, + HPXML::HVACTypeHeatPumpPTHP, + HPXML::HVACTypeHeatPumpRoom, + HPXML::HVACTypeRoomAirConditioner return HPXML::HVACCompressorTypeSingleStage end return @@ -5323,18 +5356,20 @@ def self.get_hvac_compressor_type(hvac_type, seer) # Gets the default fan power for a ceiling fan. # + # Source: ANSI/RESNET/ICC 301 + # # @return [Double] Fan power (W) def self.get_ceiling_fan_power() - # Per ANSI/RESNET/ICC 301 return 42.6 end # Gets the default quantity of ceiling fans. # + # Source: ANSI/RESNET/ICC 301 + # # @param nbeds [Integer] Number of bedrooms in the dwelling unit # @return [Integer] Number of ceiling fans def self.get_ceiling_fan_quantity(nbeds) - # Per ANSI/RESNET/ICC 301 return nbeds + 1 end @@ -5474,8 +5509,9 @@ def self.get_boiler_eae(heating_system) end end - # Gets the default interior/garage/exterior lighting fractions per ANSI/RESNET/ICC 301. - # Used by OS-ERI, OS-HEScore, etc. + # Gets the default interior/garage/exterior lighting fractions. Used by OS-ERI, OS-HEScore, etc. + # + # Source: ANSI/RESNET/ICC 301 # # @return [Hash] Map of [HPXML::LocationXXX, HPXML::LightingTypeXXX] => lighting fraction def self.get_lighting_fractions() @@ -5492,13 +5528,14 @@ def self.get_lighting_fractions() return ltg_fracs end - # Gets the default heating setpoints per ANSI/RESNET/ICC 301. + # Gets the default heating setpoints. + # + # Source: ANSI/RESNET/ICC 301 # # @param control_type [String] Thermostat control type (HPXML::HVACControlTypeXXX) # @param eri_version [String] Version of the ANSI/RESNET/ICC 301 Standard to use for equations/assumptions # @return [Array] 24 hourly comma-separated weekday and weekend setpoints def self.get_heating_setpoint(control_type, eri_version) - # Per ANSI/RESNET/ICC 301 htg_wd_setpoints = '68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68' htg_we_setpoints = '68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68, 68' if control_type == HPXML::HVACControlTypeProgrammable @@ -5515,13 +5552,14 @@ def self.get_heating_setpoint(control_type, eri_version) return htg_wd_setpoints, htg_we_setpoints end - # Gets the default cooling setpoints per ANSI/RESNET/ICC 301. + # Gets the default cooling setpoints. + # + # Source: ANSI/RESNET/ICC 301 # # @param control_type [String] Thermostat control type (HPXML::HVACControlTypeXXX) # @param eri_version [String] Version of the ANSI/RESNET/ICC 301 Standard to use for equations/assumptions # @return [Array] 24 hourly comma-separated weekday and weekend setpoints def self.get_cooling_setpoint(control_type, eri_version) - # Per ANSI/RESNET/ICC 301 clg_wd_setpoints = '78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78' clg_we_setpoints = '78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78, 78' if control_type == HPXML::HVACControlTypeProgrammable @@ -5545,24 +5583,26 @@ def self.get_cooling_setpoint(control_type, eri_version) # @return [Array] Temperature (F), heating capacity retention at the temperature (frac) def self.get_heating_capacity_retention(compressor_type, hspf = nil) retention_temp = 5.0 - if [HPXML::HVACCompressorTypeSingleStage, HPXML::HVACCompressorTypeTwoStage].include? compressor_type + case compressor_type + when HPXML::HVACCompressorTypeSingleStage, HPXML::HVACCompressorTypeTwoStage retention_fraction = 0.425 - elsif [HPXML::HVACCompressorTypeVariableSpeed].include? compressor_type + when HPXML::HVACCompressorTypeVariableSpeed # Default maximum capacity maintenance based on NEEP data for all var speed heat pump types, if not provided retention_fraction = (0.0461 * hspf + 0.1594).round(4) end return retention_temp, retention_fraction end - # Gets a 12-element array of 1s and 0s that reflects months for which the ceiling fan operates - # (i.e., when the average drybulb temperature is greater than 63F, per ANSI/RESNET/ICC 301). + # Gets the monthly ceiling fan operation schedule. + # + # Source: ANSI/RESNET/ICC 301 # # @param weather [WeatherFile] Weather object containing EPW information # @return [Array] monthly array of 1s and 0s def self.get_ceiling_fan_months(weather) months = [0] * 12 weather.data.MonthlyAvgDrybulbs.each_with_index do |val, m| - next unless val > 63.0 # F + next unless val > 63.0 # Ceiling fan operates when average drybulb temperature is greater than 63F months[m] = 1 end @@ -5628,21 +5668,21 @@ def self.get_occupancy_values() # and sensible/latent fractions. # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) - # @param num_occ [Double] Number of occupants in the dwelling unit - # @param unit_type [String] HPXML::ResidentialTypeXXX type of dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @return [Array] Plug loads annual use (kWh), sensible/latent fractions - def self.get_residual_mels_values(cfa, num_occ = nil, unit_type = nil) - if num_occ.nil? # Asset calculation + def self.get_residual_mels_values(cfa, n_occ = nil, unit_type = nil) + if n_occ.nil? # Asset calculation # ANSI/RESNET/ICC 301 annual_kwh = 0.91 * cfa else # Operational calculation # RECS 2020 if unit_type == HPXML::ResidentialTypeSFD - annual_kwh = 786.9 + 241.8 * num_occ + 0.33 * cfa + annual_kwh = 786.9 + 241.8 * n_occ + 0.33 * cfa elsif unit_type == HPXML::ResidentialTypeSFA - annual_kwh = 654.9 + 206.5 * num_occ + 0.21 * cfa + annual_kwh = 654.9 + 206.5 * n_occ + 0.21 * cfa elsif unit_type == HPXML::ResidentialTypeApartment - annual_kwh = 706.6 + 149.3 * num_occ + 0.10 * cfa + annual_kwh = 706.6 + 149.3 * n_occ + 0.10 * cfa elsif unit_type == HPXML::ResidentialTypeManufactured annual_kwh = 1795.1 # No good relationship found in RECS, so just using a constant value end @@ -5657,11 +5697,11 @@ def self.get_residual_mels_values(cfa, num_occ = nil, unit_type = nil) # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) # @param nbeds [Integer] Number of bedrooms in the dwelling unit - # @param num_occ [Double] Number of occupants in the dwelling unit - # @param unit_type [String] HPXML::ResidentialTypeXXX type of dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @return [Array] Television annual use (kWh), sensible/latent fractions - def self.get_televisions_values(cfa, nbeds, num_occ = nil, unit_type = nil) - if num_occ.nil? # Asset calculation + def self.get_televisions_values(cfa, nbeds, n_occ = nil, unit_type = nil) + if n_occ.nil? # Asset calculation # ANSI/RESNET/ICC 301 annual_kwh = 413.0 + 69.0 * nbeds else # Operational calculation @@ -5671,14 +5711,15 @@ def self.get_televisions_values(cfa, nbeds, num_occ = nil, unit_type = nil) # - SFA: 13.3 + 251.3 * num_tv # - MF: 11.4 + 250.7 * num_tv # - MH: 12.6 + 287.5 * num_tv - if unit_type == HPXML::ResidentialTypeSFD - annual_kwh = 334.0 + 92.2 * num_occ + 0.06 * cfa - elsif unit_type == HPXML::ResidentialTypeSFA - annual_kwh = 283.9 + 80.1 * num_occ + 0.07 * cfa - elsif unit_type == HPXML::ResidentialTypeApartment - annual_kwh = 190.3 + 81.0 * num_occ + 0.11 * cfa - elsif unit_type == HPXML::ResidentialTypeManufactured - annual_kwh = 99.9 + 129.6 * num_occ + 0.21 * cfa + case unit_type + when HPXML::ResidentialTypeSFD + annual_kwh = 334.0 + 92.2 * n_occ + 0.06 * cfa + when HPXML::ResidentialTypeSFA + annual_kwh = 283.9 + 80.1 * n_occ + 0.07 * cfa + when HPXML::ResidentialTypeApartment + annual_kwh = 190.3 + 81.0 * n_occ + 0.11 * cfa + when HPXML::ResidentialTypeManufactured + annual_kwh = 99.9 + 129.6 * n_occ + 0.21 * cfa end end frac_lost = 0.0 @@ -5691,29 +5732,52 @@ def self.get_televisions_values(cfa, nbeds, num_occ = nil, unit_type = nil) # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @return [Double] Annual energy use (kWh/yr) - def self.get_pool_pump_annual_energy(cfa, nbeds) - return 158.6 / 0.070 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) + def self.get_pool_pump_annual_energy(cfa, nbeds, n_occ, unit_type) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0 + end + + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + + return 158.6 / 0.070 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) end # Gets the default pool heater annual energy use. # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @param type [String] Type of heater (HPXML::HeaterTypeXXX) # @return [Array] Energy units (HPXML::UnitsXXX), annual energy use (kWh/yr or therm/yr) - def self.get_pool_heater_annual_energy(cfa, nbeds, type) + def self.get_pool_heater_annual_energy(cfa, nbeds, n_occ, unit_type, type) + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + load_units = nil load_value = nil if [HPXML::HeaterTypeElectricResistance, HPXML::HeaterTypeHeatPump].include? type load_units = HPXML::UnitsKwhPerYear - load_value = 8.3 / 0.004 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) # kWh/yr + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return load_type, 0.0 + end + + load_value = 8.3 / 0.004 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) # kWh/yr if type == HPXML::HeaterTypeHeatPump load_value /= 5.0 # Assume seasonal COP of 5.0 per https://www.energy.gov/energysaver/heat-pump-swimming-pool-heaters end elsif type == HPXML::HeaterTypeGas load_units = HPXML::UnitsThermPerYear - load_value = 3.0 / 0.014 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) # therm/yr + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return load_type, 0.0 + end + + load_value = 3.0 / 0.014 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) # therm/yr end return load_units, load_value end @@ -5722,29 +5786,52 @@ def self.get_pool_heater_annual_energy(cfa, nbeds, type) # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @return [Double] Annual energy use (kWh/yr) - def self.get_permanent_spa_pump_annual_energy(cfa, nbeds) - return 59.5 / 0.059 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) # kWh/yr + def self.get_permanent_spa_pump_annual_energy(cfa, nbeds, n_occ, unit_type) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0 + end + + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + + return 59.5 / 0.059 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) # kWh/yr end # Gets the default permanent spa heater annual energy use. # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @param type [String] Type of heater (HPXML::HeaterTypeXXX) # @return [Array] Energy units (HPXML::UnitsXXX), annual energy use (kWh/yr or therm/yr) - def self.get_permanent_spa_heater_annual_energy(cfa, nbeds, type) + def self.get_permanent_spa_heater_annual_energy(cfa, nbeds, n_occ, unit_type, type) + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + load_units = nil load_value = nil if [HPXML::HeaterTypeElectricResistance, HPXML::HeaterTypeHeatPump].include? type load_units = HPXML::UnitsKwhPerYear - load_value = 49.0 / 0.048 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) # kWh/yr + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return load_type, 0.0 + end + + load_value = 49.0 / 0.048 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) # kWh/yr if type == HPXML::HeaterTypeHeatPump load_value /= 5.0 # Assume seasonal COP of 5.0 per https://www.energy.gov/energysaver/heat-pump-swimming-pool-heaters end elsif type == HPXML::HeaterTypeGas load_units = HPXML::UnitsThermPerYear - load_value = 0.87 / 0.011 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) # therm/yr + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return load_type, 0.0 + end + + load_value = 0.87 / 0.011 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) # therm/yr end return load_units, load_value end @@ -5764,44 +5851,89 @@ def self.get_electric_vehicle_charging_annual_energy() # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @return [Double] Annual energy use (kWh/yr) - def self.get_detault_well_pump_annual_energy(cfa, nbeds) - return 50.8 / 0.127 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) + def self.get_detault_well_pump_annual_energy(cfa, nbeds, n_occ, unit_type) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0 + end + + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + + return 50.8 / 0.127 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) end # Gets the default gas grill annual energy use. # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @return [Double] Annual energy use (therm/yr) - def self.get_gas_grill_annual_energy(cfa, nbeds) - return 0.87 / 0.029 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) + def self.get_gas_grill_annual_energy(cfa, nbeds, n_occ, unit_type) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0 + end + + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + + return 0.87 / 0.029 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) end # Gets the default gas lighting annual energy use. # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @return [Double] Annual energy use (therm/yr) - def self.get_detault_gas_lighting_annual_energy(cfa, nbeds) - return 0.22 / 0.012 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) + def self.get_detault_gas_lighting_annual_energy(cfa, nbeds, n_occ, unit_type) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0 + end + + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + + return 0.22 / 0.012 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) end # Gets the default gas fireplace annual energy use. # # @param cfa [Double] Conditioned floor area in the dwelling unit (ft2) # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @return [Double] Annual energy use (therm/yr) - def self.get_gas_fireplace_annual_energy(cfa, nbeds) - return 1.95 / 0.032 * (0.5 + 0.25 * nbeds / 3.0 + 0.25 * cfa / 1920.0) + def self.get_gas_fireplace_annual_energy(cfa, nbeds, n_occ, unit_type) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0 + end + + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + + return 1.95 / 0.032 * (0.5 + 0.25 * nbeds_eq / 3.0 + 0.25 * cfa / 1920.0) end # Gets the default values associated with general water use internal gains. # - # @param nbeds_eq [Integer] Number of bedrooms (or equivalent bedrooms, as adjusted by the number of occupants) in the dwelling unit + # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @param general_water_use_usage_multiplier [Double] Usage multiplier on internal gains # @return [Array] Sensible/latent internal gains (Btu/yr) - def self.get_water_use_internal_gains(nbeds_eq, general_water_use_usage_multiplier = 1.0) + def self.get_water_use_internal_gains(nbeds, n_occ, unit_type, general_water_use_usage_multiplier = 1.0) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out internal gains + return 0.0, 0.0 + end + + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + # ANSI/RESNET/ICC 301 - Table 4.2.2(3). Internal Gains for Reference Homes sens_gains = (-1227.0 - 409.0 * nbeds_eq) * general_water_use_usage_multiplier # Btu/day lat_gains = (1245.0 + 415.0 * nbeds_eq) * general_water_use_usage_multiplier # Btu/day diff --git a/HPXMLtoOpenStudio/resources/energyplus.rb b/HPXMLtoOpenStudio/resources/energyplus.rb index 9b588d06ff..c8d5e79302 100644 --- a/HPXMLtoOpenStudio/resources/energyplus.rb +++ b/HPXMLtoOpenStudio/resources/energyplus.rb @@ -59,28 +59,25 @@ def self.fuel_type(hpxml_fuel) # Name of fuel used as inputs to E+ objects if hpxml_fuel.nil? return FuelTypeNone - elsif [HPXML::FuelTypeElectricity].include? hpxml_fuel + end + + case hpxml_fuel + when HPXML::FuelTypeElectricity return FuelTypeElectricity - elsif [HPXML::FuelTypeNaturalGas].include? hpxml_fuel + when HPXML::FuelTypeNaturalGas return FuelTypeNaturalGas - elsif [HPXML::FuelTypeOil, - HPXML::FuelTypeOil1, - HPXML::FuelTypeOil2, - HPXML::FuelTypeOil4, - HPXML::FuelTypeOil5or6, - HPXML::FuelTypeDiesel, - HPXML::FuelTypeKerosene].include? hpxml_fuel + when HPXML::FuelTypeOil, HPXML::FuelTypeOil1, HPXML::FuelTypeOil2, + HPXML::FuelTypeOil4, HPXML::FuelTypeOil5or6, HPXML::FuelTypeDiesel, + HPXML::FuelTypeKerosene return FuelTypeOil - elsif [HPXML::FuelTypePropane].include? hpxml_fuel + when HPXML::FuelTypePropane return FuelTypePropane - elsif [HPXML::FuelTypeWoodCord].include? hpxml_fuel + when HPXML::FuelTypeWoodCord return FuelTypeWoodCord - elsif [HPXML::FuelTypeWoodPellets].include? hpxml_fuel + when HPXML::FuelTypeWoodPellets return FuelTypeWoodPellets - elsif [HPXML::FuelTypeCoal, - HPXML::FuelTypeCoalAnthracite, - HPXML::FuelTypeCoalBituminous, - HPXML::FuelTypeCoke].include? hpxml_fuel + when HPXML::FuelTypeCoal, HPXML::FuelTypeCoalAnthracite, + HPXML::FuelTypeCoalBituminous, HPXML::FuelTypeCoke return FuelTypeCoal else fail "Unexpected HPXML fuel '#{hpxml_fuel}'." diff --git a/HPXMLtoOpenStudio/resources/geometry.rb b/HPXMLtoOpenStudio/resources/geometry.rb index 36965be122..cc284bbddf 100644 --- a/HPXMLtoOpenStudio/resources/geometry.rb +++ b/HPXMLtoOpenStudio/resources/geometry.rb @@ -1124,7 +1124,7 @@ def self.get_surface_z_values(surfaceArray:) # @param nbeds [Integer] Number of bedrooms in the dwelling unit # @return [Double] Number of occupants in the dwelling unit def self.get_occupancy_default_num(nbeds:) - return Float(nbeds) # Per ANSI 301 for an asset calculation + return Float(nbeds) # Per ANSI/RESNET/ICC 301 for an asset calculation end # Creates a space and zone based on contents of spaces and value of location. @@ -1325,7 +1325,6 @@ def self.create_floor_vertices(length, width, z_origin, default_azimuths) end # Set calculated zone volumes for all HPXML locations on OpenStudio Thermal Zone and Space objects. - # TODO why? for reporting? # # @param spaces [Hash] Map of HPXML locations => OpenStudio Space objects # @param hpxml_bldg [HPXML::Building] HPXML Building object representing an individual dwelling unit @@ -1640,49 +1639,50 @@ def self.calculate_zone_volume(hpxml_bldg, location) # @param location [String] the general HPXML location # @return [Hash] Map of minimum temperature, indoor/outdoor/ground weights, duct regain factor def self.get_temperature_scheduled_space_values(location) - if location == HPXML::LocationOtherHeatedSpace + case location + when HPXML::LocationOtherHeatedSpace # Average of indoor/outdoor temperatures with minimum of heating setpoint return { temp_min: 68, indoor_weight: 0.5, outdoor_weight: 0.5, ground_weight: 0.0, f_regain: 0.0 } - elsif location == HPXML::LocationOtherMultifamilyBufferSpace + when HPXML::LocationOtherMultifamilyBufferSpace # Average of indoor/outdoor temperatures with minimum of 50 F return { temp_min: 50, indoor_weight: 0.5, outdoor_weight: 0.5, ground_weight: 0.0, f_regain: 0.0 } - elsif location == HPXML::LocationOtherNonFreezingSpace + when HPXML::LocationOtherNonFreezingSpace # Floating with outdoor air temperature with minimum of 40 F return { temp_min: 40, indoor_weight: 0.0, outdoor_weight: 1.0, ground_weight: 0.0, f_regain: 0.0 } - elsif location == HPXML::LocationOtherHousingUnit + when HPXML::LocationOtherHousingUnit # Indoor air temperature return { temp_min: nil, indoor_weight: 1.0, outdoor_weight: 0.0, ground_weight: 0.0, f_regain: 0.0 } - elsif location == HPXML::LocationExteriorWall + when HPXML::LocationExteriorWall # Average of indoor/outdoor temperatures return { temp_min: nil, indoor_weight: 0.5, outdoor_weight: 0.5, ground_weight: 0.0, f_regain: 0.5 } # From LBNL's "Technical Background for default values used for Forced Air Systems in Proposed ASHRAE Standard 152P" - elsif location == HPXML::LocationUnderSlab + when HPXML::LocationUnderSlab # Ground temperature return { temp_min: nil, indoor_weight: 0.0, outdoor_weight: 0.0, ground_weight: 1.0, f_regain: 0.83 } # From LBNL's "Technical Background for default values used for Forced Air Systems in Proposed ASHRAE Standard 152P" - elsif location == HPXML::LocationManufacturedHomeBelly + when HPXML::LocationManufacturedHomeBelly # From LBNL's "Technical Background for default values used for Forced Air Systems in Proposed ASHRAE Standard 152P" # 3.5 Manufactured House Belly Pan Temperatures # FUTURE: Consider modeling the belly as a separate thermal zone so that we dynamically calculate temperatures. diff --git a/HPXMLtoOpenStudio/resources/hotwater_appliances.rb b/HPXMLtoOpenStudio/resources/hotwater_appliances.rb index 6a6182a223..36dbfcb758 100644 --- a/HPXMLtoOpenStudio/resources/hotwater_appliances.rb +++ b/HPXMLtoOpenStudio/resources/hotwater_appliances.rb @@ -21,8 +21,8 @@ def self.apply(runner, model, weather, spaces, hpxml_bldg, hpxml_header, schedul fixtures_usage_multiplier = hpxml_bldg.water_heating.water_fixtures_usage_multiplier conditioned_space = spaces[HPXML::LocationConditionedSpace] nbeds = hpxml_bldg.building_construction.number_of_bedrooms - nbeds_eq = hpxml_bldg.building_construction.additional_properties.equivalent_number_of_bedrooms n_occ = hpxml_bldg.building_occupancy.number_of_residents + unit_type = hpxml_bldg.building_construction.residential_facility_type eri_version = hpxml_header.eri_calculation_version unit_multiplier = hpxml_bldg.building_construction.number_of_units @@ -56,7 +56,7 @@ def self.apply(runner, model, weather, spaces, hpxml_bldg, hpxml_header, schedul # Clothes washer energy if not clothes_washer.nil? cw_space = Geometry.get_space_from_location(clothes_washer.location, spaces) - cw_annual_kwh, cw_frac_sens, cw_frac_lat, cw_gpd = calc_clothes_washer_energy_gpd(runner, eri_version, nbeds, clothes_washer, cw_space.nil?, n_occ) + cw_annual_kwh, cw_frac_sens, cw_frac_lat, cw_gpd = calc_clothes_washer_energy_gpd(runner, eri_version, nbeds, n_occ, clothes_washer, cw_space.nil?) # Create schedule cw_power_schedule = nil @@ -98,7 +98,7 @@ def self.apply(runner, model, weather, spaces, hpxml_bldg, hpxml_header, schedul # Clothes dryer energy if not clothes_dryer.nil? cd_space = Geometry.get_space_from_location(clothes_dryer.location, spaces) - cd_annual_kwh, cd_annual_therm, cd_frac_sens, cd_frac_lat = calc_clothes_dryer_energy(runner, eri_version, nbeds, clothes_dryer, clothes_washer, cd_space.nil?, n_occ) + cd_annual_kwh, cd_annual_therm, cd_frac_sens, cd_frac_lat = calc_clothes_dryer_energy(runner, eri_version, nbeds, n_occ, clothes_dryer, clothes_washer, cd_space.nil?) # Create schedule cd_schedule = nil @@ -154,7 +154,7 @@ def self.apply(runner, model, weather, spaces, hpxml_bldg, hpxml_header, schedul # Dishwasher energy if not dishwasher.nil? dw_space = Geometry.get_space_from_location(dishwasher.location, spaces) - dw_annual_kwh, dw_frac_sens, dw_frac_lat, dw_gpd = calc_dishwasher_energy_gpd(runner, eri_version, nbeds, dishwasher, dw_space.nil?, n_occ) + dw_annual_kwh, dw_frac_sens, dw_frac_lat, dw_gpd = calc_dishwasher_energy_gpd(runner, eri_version, nbeds, n_occ, dishwasher, dw_space.nil?) # Create schedule dw_power_schedule = nil @@ -300,7 +300,7 @@ def self.apply(runner, model, weather, spaces, hpxml_bldg, hpxml_header, schedul # Cooking Range energy if not cooking_range.nil? cook_space = Geometry.get_space_from_location(cooking_range.location, spaces) - cook_annual_kwh, cook_annual_therm, cook_frac_sens, cook_frac_lat = calc_range_oven_energy(runner, nbeds_eq, cooking_range, oven, cook_space.nil?) + cook_annual_kwh, cook_annual_therm, cook_frac_sens, cook_frac_lat = calc_range_oven_energy(runner, nbeds, n_occ, unit_type, cooking_range, oven, cook_space.nil?) # Create schedule cook_schedule = nil @@ -390,7 +390,7 @@ def self.apply(runner, model, weather, spaces, hpxml_bldg, hpxml_header, schedul wh_setpoint = Defaults.get_water_heater_temperature(eri_version) if wh_setpoint.nil? # using detailed schedules avg_setpoint_temp += wh_setpoint * water_heating_system.fraction_dhw_load_served end - daily_wh_inlet_temperatures = calc_water_heater_daily_inlet_temperatures(weather, nbeds_eq, hot_water_distribution, frac_low_flow_fixtures) + daily_wh_inlet_temperatures = calc_water_heater_daily_inlet_temperatures(weather, nbeds, n_occ, unit_type, hot_water_distribution, frac_low_flow_fixtures) daily_wh_inlet_temperatures_c = daily_wh_inlet_temperatures.map { |t| UnitConversions.convert(t, 'F', 'C') } daily_mw_fractions = calc_mixed_water_daily_fractions(daily_wh_inlet_temperatures, avg_setpoint_temp, t_mix) @@ -589,12 +589,21 @@ def self.apply(runner, model, weather, spaces, hpxml_bldg, hpxml_header, schedul # Calculates cooking range/oven annual energy use. # # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings - # @param nbeds_eq [Integer] Number of bedrooms (or equivalent bedrooms, as adjusted by the number of occupants) in the dwelling unit + # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @param cooking_range [HPXML::CookingRange] The HPXML cooking range of interest # @param oven [HPXML::Oven] The HPXML oven of interest # @param is_outside [Boolean] Whether the appliance is located outside the dwelling unit # @return [Array] Annual electricity use (kWh), annual fuel use (therm), sensible/latent fractions - def self.calc_range_oven_energy(runner, nbeds_eq, cooking_range, oven, is_outside = false) + def self.calc_range_oven_energy(runner, nbeds, n_occ, unit_type, cooking_range, oven, is_outside = false) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0, 0.0, 0.0, 0.0 + end + + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + if cooking_range.is_induction burner_ef = 0.91 else @@ -644,11 +653,16 @@ def self.calc_range_oven_energy(runner, nbeds_eq, cooking_range, oven, is_outsid # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings # @param eri_version [String] Version of the ANSI/RESNET/ICC 301 Standard to use for equations/assumptions # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit # @param dishwasher [HPXML::Dishwasher] The HPXML dishwasher of interest # @param is_outside [Boolean] Whether the appliance is located outside the dwelling unit - # @param n_occ [Double] Number of occupants in the dwelling unit # @return [Array] Annual electricity use (kWh), sensible/latent fractions, hot water use (gal/day) - def self.calc_dishwasher_energy_gpd(runner, eri_version, nbeds, dishwasher, is_outside = false, n_occ = nil) + def self.calc_dishwasher_energy_gpd(runner, eri_version, nbeds, n_occ, dishwasher, is_outside = false) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0, 0.0, 0.0, 0.0 + end + if Constants::ERIVersions.index(eri_version) >= Constants::ERIVersions.index('2019A') if dishwasher.rated_annual_kwh.nil? dishwasher.rated_annual_kwh = calc_dishwasher_annual_kwh_from_ef(dishwasher.energy_factor) @@ -700,21 +714,23 @@ def self.calc_dishwasher_energy_gpd(runner, eri_version, nbeds, dishwasher, is_o return annual_kwh, frac_sens, frac_lat, gpd end - # Converts dishwasher rated annual use (kWh) to energy factor (EF). + # Calculates dishwasher rated energy factor (EF) from annual use (kWh). + # + # Source: ANSI/RESNET/ICC 301 # # @param annual_kwh [Double] Rated annual kWh # @return [Double] Energy factor def self.calc_dishwasher_ef_from_annual_kwh(annual_kwh) - # Per ANSI/RESNET/ICC 301 return 215.0 / annual_kwh end - # Converts dishwasher energy factor (EF) to rated annual use (kWh). + # Calculates dishwasher annual use (kWh) from rated energy factor (EF). + # + # Source: ANSI/RESNET/ICC 301 # # @param ef [Double] Energy factor # @return [Double] Rated annual use (kWh) def self.calc_dishwasher_annual_kwh_from_ef(ef) - # Per ANSI/RESNET/ICC 301 return 215.0 / ef end @@ -723,12 +739,17 @@ def self.calc_dishwasher_annual_kwh_from_ef(ef) # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings # @param eri_version [String] Version of the ANSI/RESNET/ICC 301 Standard to use for equations/assumptions # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit # @param clothes_dryer [HPXML::ClothesDryer] The HPXML clothes dryer of interest # @param clothes_washer [HPXML::ClothesWasher] The related HPXML clothes washer, which affects dryer use # @param is_outside [Boolean] Whether the appliance is located outside the dwelling unit - # @param n_occ [Double] Number of occupants in the dwelling unit # @return [Array] Annual electricity use (kWh), annual fuel use (therm), sensible/latent fractions - def self.calc_clothes_dryer_energy(runner, eri_version, nbeds, clothes_dryer, clothes_washer, is_outside = false, n_occ = nil) + def self.calc_clothes_dryer_energy(runner, eri_version, nbeds, n_occ, clothes_dryer, clothes_washer, is_outside = false) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0, 0.0, 0.0, 0.0 + end + if Constants::ERIVersions.index(eri_version) >= Constants::ERIVersions.index('2019A') if clothes_dryer.combined_energy_factor.nil? clothes_dryer.combined_energy_factor = calc_clothes_dryer_cef_from_ef(clothes_dryer.energy_factor) @@ -796,15 +817,23 @@ def self.calc_clothes_dryer_energy(runner, eri_version, nbeds, clothes_dryer, cl return annual_kwh, annual_therm, frac_sens, frac_lat end - # Converts clothes dryer energy factor (EF) to combined energy factor (CEF). + # Calculates clothes dryer combined energy factor (CEF) from energy factor (EF). + # + # Source: RESNET's Interpretation on Clothes Dryer CEF + # https://www.resnet.us/wp-content/uploads/No.-301-2014-10-Section-4.2.2.5.2.8-Clothes-Dryer-CEF-Rating.pdf + # Note that this is a regression based on products on the market, not a conversion. # # @param ef [Double] Energy factor # @return [Double] Combined energy factor def self.calc_clothes_dryer_cef_from_ef(ef) - return ef / 1.15 # Interpretation on ANSI/RESNET/ICC 301-2014 Clothes Dryer CEF + return ef / 1.15 end - # Converts clothes dryer combined energy factor (CEF) to energy factor (EF). + # Calculates clothes dryer energy factor (EF) from combined energy factor (CEF). + # + # Source: RESNET's Interpretation on Clothes Dryer CEF + # https://www.resnet.us/wp-content/uploads/No.-301-2014-10-Section-4.2.2.5.2.8-Clothes-Dryer-CEF-Rating.pdf + # Note that this is a regression based on products on the market, not a conversion. # # @param cef [Double] Combined energy factor # @return [Double] Energy factor @@ -817,11 +846,16 @@ def self.calc_clothes_dryer_ef_from_cef(cef) # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings # @param eri_version [String] Version of the ANSI/RESNET/ICC 301 Standard to use for equations/assumptions # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit # @param clothes_washer [HPXML::ClothesWasher] The HPXML clothes washer of interest # @param is_outside [Boolean] Whether the appliance is located outside the dwelling unit - # @param n_occ [Double] Number of occupants in the dwelling unit # @return [Array] Annual electricity use (kWh), sensible/latent fractions, hot water use (gal/day) - def self.calc_clothes_washer_energy_gpd(runner, eri_version, nbeds, clothes_washer, is_outside = false, n_occ = nil) + def self.calc_clothes_washer_energy_gpd(runner, eri_version, nbeds, n_occ, clothes_washer, is_outside = false) + if n_occ == 0 + # Operational calculation w/ zero occupants, zero out energy use + return 0.0, 0.0, 0.0, 0.0 + end + if Constants::ERIVersions.index(eri_version) >= Constants::ERIVersions.index('2019A') gas_h20 = 0.3914 # (gal/cyc) per (therm/y) elec_h20 = 0.0178 # (gal/cyc) per (kWh/y) @@ -869,20 +903,28 @@ def self.calc_clothes_washer_energy_gpd(runner, eri_version, nbeds, clothes_wash return annual_kwh, frac_sens, frac_lat, gpd end - # Converts clothes washer modified energy factor (MEF) to integrated modified energy factor (IMEF). + # Calculates clothes washer integrated modified energy factor (IMEF) from modified energy factor (MEF). + # + # Source: RESNET's Interpretation on Clothes Washer IMEF + # https://www.resnet.us/wp-content/uploads/No.-301-2014-08-sECTION-4.2.2.5.2.8-Clothes-Washers-Eq-4.2-6.pdf + # Note that this is a regression based on products on the market, not a conversion. # # @param mef [Double] Modified energy factor # @return [Double] Integrated modified energy factor def self.calc_clothes_washer_imef_from_mef(mef) - return (mef - 0.503) / 0.95 # Interpretation on ANSI/RESNET 301-2014 Clothes Washer IMEF + return (mef - 0.503) / 0.95 end - # Converts clothes washer integrated modified energy factor (IMEF) to modified energy factor (MEF). + # Calculates clothes washer modified energy factor (MEF) from integrated modified energy factor (IMEF). # - # @param mef [Double] Modified energy factor - # @return [Double] Integrated modified energy factor + # Source: RESNET's Interpretation on Clothes Washer IMEF + # https://www.resnet.us/wp-content/uploads/No.-301-2014-08-sECTION-4.2.2.5.2.8-Clothes-Washers-Eq-4.2-6.pdf + # Note that this is a regression based on products on the market, not a conversion. + # + # @param imef [Double] Integrated modified energy factor + # @return [Double] Modified energy factor def self.calc_clothes_washer_mef_from_imef(imef) - return 0.503 + 0.95 * imef # Interpretation on ANSI/RESNET 301-2014 Clothes Washer IMEF + return 0.503 + 0.95 * imef # Interpretation on ANSI/RESNET/ICC 301-2014 Clothes Washer IMEF end # Calculates refrigerator annual energy use. @@ -995,20 +1037,20 @@ def self.get_fridge_or_freezer_coefficients_schedule(model, col_name, obj_name, return schedule end - # Calculates Drain Water Heat Recovery (DWHR) factors per ANSI/RESNET/ICC 301. + # Calculates Drain Water Heat Recovery (DWHR) factors. + # + # Source: ANSI/RESNET/ICC 301 # # @param nbeds_eq [Integer] Number of bedrooms (or equivalent bedrooms, as adjusted by the number of occupants) in the dwelling unit # @param hot_water_distribution [HPXML::HotWaterDistribution] The HPXML hot water distribution system of interest # @param frac_low_flow_fixtures [Double] The fraction of fixtures considered low-flow # @return [Array] Effectiveness (frac), fraction of water impacted by DWHR, piping loss coefficient, location factor, fixture factor def self.get_dwhr_factors(nbeds_eq, hot_water_distribution, frac_low_flow_fixtures) - # ANSI/RESNET 301-2014 Addendum A-2015 - # Amendment on Domestic Hot Water (DHW) Systems - # Eq. 4.2-14 + # ANSI/RESNET/ICC 301-2022 Eq. 4.2-42 eff_adj = 1.0 + 0.082 * frac_low_flow_fixtures - iFrac = 0.56 + 0.015 * nbeds_eq - 0.0004 * nbeds_eq**2 # fraction of hot water use impacted by DWHR + i_frac = 0.56 + 0.015 * nbeds_eq - 0.0004 * nbeds_eq**2 # fraction of hot water use impacted by DWHR if hot_water_distribution.system_type == HPXML::DHWDistTypeRecirc pLength = hot_water_distribution.recirculation_branch_piping_length @@ -1019,30 +1061,38 @@ def self.get_dwhr_factors(nbeds_eq, hot_water_distribution, frac_low_flow_fixtur # Location factors for DWHR placement if hot_water_distribution.dwhr_equal_flow - locF = 1.000 + loc_f = 1.000 else - locF = 0.777 + loc_f = 0.777 end # Fixture Factor if hot_water_distribution.dwhr_facilities_connected == HPXML::DWHRFacilitiesConnectedAll - fixF = 1.0 + fix_f = 1.0 elsif hot_water_distribution.dwhr_facilities_connected == HPXML::DWHRFacilitiesConnectedOne - fixF = 0.5 + fix_f = 0.5 end - return eff_adj, iFrac, plc, locF, fixF + return eff_adj, i_frac, plc, loc_f, fix_f end # Calculates daily water heater inlet temperatures, which includes an adjustment if # there is a drain water heat recovery device. # # @param weather [WeatherFile] Weather object containing EPW information - # @param nbeds_eq [Integer] Number of bedrooms (or equivalent bedrooms, as adjusted by the number of occupants) in the dwelling unit + # @param nbeds [Integer] Number of bedrooms in the dwelling unit + # @param n_occ [Double] Number of occupants in the dwelling unit + # @param unit_type [String] Type of dwelling unit (HXPML::ResidentialTypeXXX) # @param hot_water_distribution [HPXML::HotWaterDistribution] The HPXML hot water distribution system of interest # @param frac_low_flow_fixtures [Double] The fraction of fixtures considered low-flow # @return [Array] Daily water heater inlet temperatures (F) - def self.calc_water_heater_daily_inlet_temperatures(weather, nbeds_eq, hot_water_distribution, frac_low_flow_fixtures) + def self.calc_water_heater_daily_inlet_temperatures(weather, nbeds, n_occ, unit_type, hot_water_distribution, frac_low_flow_fixtures) + if n_occ == 0 + # Operational calculation w/ zero occupants + nbeds_eq = 0 + else + nbeds_eq = Defaults.get_equivalent_nbeds(nbeds, n_occ, unit_type) + end wh_temps_daily = weather.data.MainsDailyTemps.dup if (not hot_water_distribution.dwhr_efficiency.nil?) # Per ANSI/RESNET/ICC 301 @@ -1093,14 +1143,14 @@ def self.get_hwdist_recirc_pump_energy(hot_water_distribution, fixtures_usage_mu # Annual electricity consumption factor for hot water recirculation system pumps # Assume the fixtures_usage_multiplier only applies for Sensor/Manual control type. if hot_water_distribution.system_type == HPXML::DHWDistTypeRecirc - if [HPXML::DHWRecircControlTypeNone, - HPXML::DHWRecircControlTypeTimer].include? hot_water_distribution.recirculation_control_type + case hot_water_distribution.recirculation_control_type + when HPXML::DHWRecircControlTypeNone, HPXML::DHWRecircControlTypeTimer dist_pump_annual_kwh += (8.76 * hot_water_distribution.recirculation_pump_power) - elsif [HPXML::DHWRecircControlTypeTemperature].include? hot_water_distribution.recirculation_control_type + when HPXML::DHWRecircControlTypeTemperature dist_pump_annual_kwh += (1.46 * hot_water_distribution.recirculation_pump_power) - elsif [HPXML::DHWRecircControlTypeSensor].include? hot_water_distribution.recirculation_control_type + when HPXML::DHWRecircControlTypeSensor dist_pump_annual_kwh += (0.15 * hot_water_distribution.recirculation_pump_power * fixtures_usage_multiplier) - elsif [HPXML::DHWRecircControlTypeManual].include? hot_water_distribution.recirculation_control_type + when HPXML::DHWRecircControlTypeManual dist_pump_annual_kwh += (0.10 * hot_water_distribution.recirculation_pump_power * fixtures_usage_multiplier) else fail "Unexpected hot water distribution system recirculation type: '#{hot_water_distribution.recirculation_control_type}'." @@ -1115,12 +1165,10 @@ def self.get_hwdist_recirc_pump_energy(hot_water_distribution, fixtures_usage_mu # Assume the fixtures_usage_multiplier only applies for Sensor/Manual control type. if hot_water_distribution.has_shared_recirculation n_bdeq = hot_water_distribution.shared_recirculation_number_of_bedrooms_served - if [HPXML::DHWRecircControlTypeNone, - HPXML::DHWRecircControlTypeTimer, - HPXML::DHWRecircControlTypeTemperature].include? hot_water_distribution.shared_recirculation_control_type + case hot_water_distribution.shared_recirculation_control_type + when HPXML::DHWRecircControlTypeNone, HPXML::DHWRecircControlTypeTimer, HPXML::DHWRecircControlTypeTemperature op_hrs = 8760.0 - elsif [HPXML::DHWRecircControlTypeSensor, - HPXML::DHWRecircControlTypeManual].include? hot_water_distribution.shared_recirculation_control_type + when HPXML::DHWRecircControlTypeSensor, HPXML::DHWRecircControlTypeManual op_hrs = 730.0 * fixtures_usage_multiplier else fail "Unexpected hot water distribution system shared recirculation type: '#{hot_water_distribution.shared_recirculation_control_type}'." @@ -1153,10 +1201,8 @@ def self.get_fixtures_effectiveness(frac_low_flow_fixtures) # @return [Double] Mixed water use (gal/day) def self.get_fixtures_gpd(eri_version, nbeds, frac_low_flow_fixtures, daily_mw_fractions, fixtures_usage_multiplier = 1.0, n_occ = nil) if Constants::ERIVersions.index(eri_version) >= Constants::ERIVersions.index('2014A') - # ANSI/RESNET 301-2014 Addendum A-2015 - # Amendment on Domestic Hot Water (DHW) Systems if n_occ.nil? # Asset calculation - ref_f_gpd = 14.6 + 10.0 * nbeds # Eq. 4.2-2 (refFgpd) + ref_f_gpd = 14.6 + 10.0 * nbeds # ANSI/RESNET/ICC 301-2022 Eq. 4.2-29 (refFgpd) else # Operational calculation ref_f_gpd = [-4.84 + 18.6 * n_occ, 0.0].max # Eq. 14 from http://www.fsec.ucf.edu/en/publications/pdf/fsec-pf-464-15.pdf end @@ -1172,6 +1218,8 @@ def self.get_fixtures_gpd(eri_version, nbeds, frac_low_flow_fixtures, daily_mw_f # Calculates the equivalent daily mixed (not hot) water use associated with the distribution system. # + # Source: ANSI/RESNET/ICC 301 + # # @param eri_version [String] Version of the ANSI/RESNET/ICC 301 Standard to use for equations/assumptions # @param nbeds [Integer] Number of bedrooms in the dwelling unit # @param has_uncond_bsmnt [Boolean] Whether the dwelling unit has an unconditioned basement @@ -1189,33 +1237,37 @@ def self.get_dist_waste_gpd(eri_version, nbeds, has_uncond_bsmnt, has_cond_bsmnt return 0.0 end - # ANSI/RESNET 301-2014 Addendum A-2015 - # Amendment on Domestic Hot Water (DHW) Systems - # 4.2.2.5.2.11 Service Hot Water Use + # ANSI/RESNET/ICC 301-2022 Section 4.2.2.7.1.4 - # Table 4.2.2.5.2.11(2) Hot Water Distribution System Insulation Factors + # Table 4.2.2.7.2.11(2) Hot Water Distribution System Insulation Factors sys_factor = nil - if (hot_water_distribution.system_type == HPXML::DHWDistTypeRecirc) && (hot_water_distribution.pipe_r_value < 3.0) - sys_factor = 1.11 - elsif (hot_water_distribution.system_type == HPXML::DHWDistTypeRecirc) && (hot_water_distribution.pipe_r_value >= 3.0) - sys_factor = 1.0 - elsif (hot_water_distribution.system_type == HPXML::DHWDistTypeStandard) && (hot_water_distribution.pipe_r_value >= 3.0) - sys_factor = 0.90 - elsif (hot_water_distribution.system_type == HPXML::DHWDistTypeStandard) && (hot_water_distribution.pipe_r_value < 3.0) - sys_factor = 1.0 + case hot_water_distribution.system_type + when HPXML::DHWDistTypeRecirc + if hot_water_distribution.pipe_r_value < 3.0 + sys_factor = 1.11 + elsif hot_water_distribution.pipe_r_value >= 3.0 + sys_factor = 1.0 + end + when HPXML::DHWDistTypeStandard + if hot_water_distribution.pipe_r_value >= 3.0 + sys_factor = 0.90 + elsif hot_water_distribution.pipe_r_value < 3.0 + sys_factor = 1.0 + end end if n_occ.nil? # Asset calculation - ref_w_gpd = 9.8 * (nbeds**0.43) # Eq. 4.2-2 (refWgpd) + ref_w_gpd = 9.8 * (nbeds**0.43) # ANSI/RESNET/ICC 301-2022 Eq. 4.2-29 (refWgpd) else # Operational calculation ref_w_gpd = 7.16 * (n_occ**0.7) # Eq. 14 from http://www.fsec.ucf.edu/en/publications/pdf/fsec-pf-464-15.pdf end o_frac = 0.25 o_cd_eff = 0.0 - if hot_water_distribution.system_type == HPXML::DHWDistTypeRecirc + case hot_water_distribution.system_type + when HPXML::DHWDistTypeRecirc p_ratio = hot_water_distribution.recirculation_branch_piping_length / 10.0 - elsif hot_water_distribution.system_type == HPXML::DHWDistTypeStandard + when HPXML::DHWDistTypeStandard ref_pipe_l = Defaults.get_std_pipe_length(has_uncond_bsmnt, has_cond_bsmnt, cfa, ncfl) p_ratio = hot_water_distribution.standard_piping_length / ref_pipe_l end @@ -1224,9 +1276,10 @@ def self.get_dist_waste_gpd(eri_version, nbeds, has_uncond_bsmnt, has_cond_bsmnt s_w_gpd = (ref_w_gpd - ref_w_gpd * o_frac) * p_ratio * sys_factor # Eq. 4.2-13 # Table 4.2.2.5.2.11(3) Distribution system water use effectiveness - if hot_water_distribution.system_type == HPXML::DHWDistTypeRecirc + case hot_water_distribution.system_type + when HPXML::DHWDistTypeRecirc wd_eff = 0.1 - elsif hot_water_distribution.system_type == HPXML::DHWDistTypeStandard + when HPXML::DHWDistTypeStandard wd_eff = 1.0 end diff --git a/HPXMLtoOpenStudio/resources/hpxml.rb b/HPXMLtoOpenStudio/resources/hpxml.rb index aa0242e034..b062fa4ca1 100644 --- a/HPXMLtoOpenStudio/resources/hpxml.rb +++ b/HPXMLtoOpenStudio/resources/hpxml.rb @@ -660,40 +660,62 @@ def set_unique_hpxml_ids(hpxml_doc, last_building_only = false) # Returns a hash with whether each fuel exists in the HPXML Building or Buildings # - # @param hpxml_doc [Oga::XML::Document] HPXML object as an XML document # @param building_id [String] If provided, only search the single HPXML Building with the given ID # @return [Hash] Map of HPXML::FuelTypeXXX => boolean - def has_fuels(hpxml_doc, building_id = nil) - has_fuels = {} - - fuel_element_names = ['HeatingSystemFuel', - 'CoolingSystemFuel', - 'HeatPumpFuel', - 'BackupSystemFuel', - 'FuelType', - 'IntegratedHeatingSystemFuel', - 'Heater/Type'] + def has_fuels(building_id = nil) + has_fuel = {} + has_fuel[HPXML::FuelTypeElectricity] = true HPXML::fossil_fuels.each do |fuel| - has_fuels[fuel] = false - fuel_element_names.each do |fuel_element_name| - if fuel_element_name == 'Heater/Type' && fuel == HPXML::FuelTypeNaturalGas - fuel_element_value = HPXML::HeaterTypeGas - else - fuel_element_value = fuel + has_fuel[fuel] = false + + buildings.each do |hpxml_bldg| + next if (not building_id.nil?) && (hpxml_bldg.building_id != building_id) + + # Check HVAC systems + hpxml_bldg.hvac_systems.each do |hvac_system| + if hvac_system.respond_to?(:heating_system_fuel) && hvac_system.heating_system_fuel == fuel + has_fuel[fuel] = true + end + if hvac_system.respond_to?(:cooling_system_fuel) && hvac_system.cooling_system_fuel == fuel + has_fuel[fuel] = true + end + if hvac_system.respond_to?(:heat_pump_fuel) && hvac_system.heat_pump_fuel == fuel + has_fuel[fuel] = true + end + if hvac_system.respond_to?(:backup_heating_fuel) && hvac_system.backup_heating_fuel == fuel + has_fuel[fuel] = true + end + if hvac_system.respond_to?(:integrated_heating_system_fuel) && hvac_system.integrated_heating_system_fuel == fuel + has_fuel[fuel] = true + end end - search_str = "/HPXML/Building[BuildingID/@id='#{building_id}']//#{fuel_element_name}[text() = '#{fuel_element_value}']" - if building_id.nil? - search_str = "/HPXML/Building//#{fuel_element_name}[text() = '#{fuel_element_value}']" + + # Check other appliances + (hpxml_bldg.water_heating_systems + + hpxml_bldg.generators + + hpxml_bldg.clothes_dryers + + hpxml_bldg.cooking_ranges + + hpxml_bldg.fuel_loads).each do |appliance| + if appliance.fuel_type == fuel + has_fuel[fuel] = true + end end - if XMLHelper.has_element(hpxml_doc, search_str) - has_fuels[fuel] = true - break + + # Check pool/spa heaters + if fuel == HPXML::FuelTypeNaturalGas + (hpxml_bldg.pools + hpxml_bldg.permanent_spas).each do |pool_or_spa| + if pool_or_spa.heater_type == HPXML::HeaterTypeGas + has_fuel[fuel] = true + end + end end + + break if has_fuel[fuel] end end - return has_fuels + return has_fuel end # Object to store additional properties on an HPXML object that are not intended @@ -1714,11 +1736,9 @@ def has_fuel_access # Returns a hash with whether each fuel exists in the HPXML Building. # - # @param hpxml_doc [Oga::XML::Document] HPXML object as an XML document # @return [Hash] Map of HPXML::FuelTypeXXX => boolean - def has_fuels(hpxml_doc) - # Returns a hash with whether each fuel exists in the HPXML Building - return @parent_object.has_fuels(hpxml_doc, @building_id) + def has_fuels() + return @parent_object.has_fuels(@building_id) end # Returns the predominant heating fuel type (weighted by fraction of @@ -1828,7 +1848,7 @@ def has_walkout_basement end # Calculates above-grade and below-grade thermal boundary wall areas. - # Used to calculate the window area in the ERI Reference Home per ANSI 301. + # Used to calculate the window area in the ERI Reference Home. # # Thermal boundary wall is any wall that separates conditioned space from # unconditioned space, outside, or soil. Above-grade thermal boundary @@ -1836,6 +1856,8 @@ def has_walkout_basement # Below-grade thermal boundary wall is any portion of a thermal boundary # wall in contact with soil. # + # Source: ANSI/RESNET/ICC 301 + # # @return [Array] Above-grade and below-grade thermal boundary wall areas (ft2) def thermal_boundary_wall_areas ag_wall_area = 0.0 @@ -1881,8 +1903,9 @@ def above_grade_conditioned_volume return ag_cond_vol end - # Calculates common wall area. - # Used to calculate the window area in the ERI Reference Home per ANSI 301. + # Calculates common wall area. Used to calculate the window area in the ERI Reference Home. + # + # Source: ANSI/RESNET/ICC 301 # # Common wall is the total wall area of walls adjacent to other unit's # conditioned space, not including foundation walls. @@ -1904,12 +1927,14 @@ def common_wall_area # Returns the total and exterior compartmentalization boundary area. # Used to convert between total infiltration and exterior infiltration for - # SFA/MF dwelling units per ANSI 301. + # SFA/MF dwelling units. + # + # Source: ANSI/RESNET/ICC 301 # # @return [Array] Total and exterior compartmentalization areas (ft2) def compartmentalization_boundary_areas total_area = 0.0 # Total surface area that bounds the Infiltration Volume - exterior_area = 0.0 # Same as above excluding surfaces attached to garage, other housing units, or other multifamily spaces (see 301-2019 Addendum B) + exterior_area = 0.0 # Same as above excluding surfaces attached to garage, other housing units, or other multifamily spaces # Determine which spaces are within infiltration volume spaces_within_infil_volume = HPXML::conditioned_locations_this_unit @@ -2699,11 +2724,9 @@ def check_for_errors # @param building [Oga::XML::Element] The current Building XML element # @return [nil] def to_doc(building) - return if nil? - climate_and_risk_zones = XMLHelper.create_elements_as_needed(building, ['BuildingDetails', 'ClimateandRiskZones']) - @climate_zone_ieccs.to_doc(climate_and_risk_zones) + return if nil? if not @weather_station_id.nil? weather_station = XMLHelper.add_element(climate_and_risk_zones, 'WeatherStation') @@ -3407,11 +3430,12 @@ def attached_floors def to_location return if @attic_type.nil? - if [AtticTypeCathedral, AtticTypeConditioned, AtticTypeFlatRoof, AtticTypeBelowApartment].include? @attic_type + case @attic_type + when AtticTypeCathedral, AtticTypeConditioned, AtticTypeFlatRoof, AtticTypeBelowApartment return LocationConditionedSpace - elsif [AtticTypeUnvented].include? @attic_type + when AtticTypeUnvented return LocationAtticUnvented - elsif [AtticTypeVented].include? @attic_type + when AtticTypeVented return LocationAtticVented else fail "Unexpected attic type: '#{@attic_type}'." @@ -3450,12 +3474,13 @@ def to_doc(building) XMLHelper.add_attribute(sys_id, 'id', @id) if not @attic_type.nil? attic_type_el = XMLHelper.add_element(attic, 'AtticType') - if [AtticTypeFlatRoof, AtticTypeCathedral, AtticTypeBelowApartment].include? @attic_type + case @attic_type + when AtticTypeFlatRoof, AtticTypeCathedral, AtticTypeBelowApartment XMLHelper.add_element(attic_type_el, @attic_type) - elsif [AtticTypeUnvented].include? @attic_type + when AtticTypeUnvented attic_type_attic = XMLHelper.add_element(attic_type_el, 'Attic') XMLHelper.add_element(attic_type_attic, 'Vented', false, :boolean) - elsif [AtticTypeVented].include? @attic_type + when AtticTypeVented attic_type_attic = XMLHelper.add_element(attic_type_el, 'Attic') XMLHelper.add_element(attic_type_attic, 'Vented', true, :boolean) if not @vented_attic_sla.nil? @@ -3467,7 +3492,7 @@ def to_doc(building) XMLHelper.add_element(ventilation_rate, 'UnitofMeasure', UnitsACHNatural, :string) XMLHelper.add_element(ventilation_rate, 'Value', @vented_attic_ach, :float) end - elsif [AtticTypeConditioned].include? @attic_type + when AtticTypeConditioned attic_type_attic = XMLHelper.add_element(attic_type_el, 'Attic') XMLHelper.add_element(attic_type_attic, 'Conditioned', true, :boolean) else @@ -3648,21 +3673,22 @@ def attached_rim_joists def to_location return if @foundation_type.nil? - if [FoundationTypeSlab, FoundationTypeAboveApartment].include? @foundation_type + case @foundation_type + when FoundationTypeSlab, FoundationTypeAboveApartment return LocationConditionedSpace - elsif [FoundationTypeAmbient].include? @foundation_type + when FoundationTypeAmbient return LocationOutside - elsif [FoundationTypeBasementConditioned].include? @foundation_type + when FoundationTypeBasementConditioned return LocationBasementConditioned - elsif [FoundationTypeBasementUnconditioned].include? @foundation_type + when FoundationTypeBasementUnconditioned return LocationBasementUnconditioned - elsif [FoundationTypeCrawlspaceUnvented].include? @foundation_type + when FoundationTypeCrawlspaceUnvented return LocationCrawlspaceUnvented - elsif [FoundationTypeCrawlspaceVented].include? @foundation_type + when FoundationTypeCrawlspaceVented return LocationCrawlspaceVented - elsif @foundation_type == FoundationTypeCrawlspaceConditioned + when FoundationTypeCrawlspaceConditioned return LocationCrawlspaceConditioned - elsif @foundation_type == FoundationTypeBellyAndWing + when FoundationTypeBellyAndWing return LocationManufacturedHomeUnderBelly else fail "Unexpected foundation type: '#{@foundation_type}'." @@ -3721,15 +3747,16 @@ def to_doc(building) XMLHelper.add_attribute(sys_id, 'id', @id) if not @foundation_type.nil? foundation_type_el = XMLHelper.add_element(foundation, 'FoundationType') - if [FoundationTypeSlab, FoundationTypeAmbient, FoundationTypeAboveApartment].include? @foundation_type + case @foundation_type + when FoundationTypeSlab, FoundationTypeAmbient, FoundationTypeAboveApartment XMLHelper.add_element(foundation_type_el, @foundation_type) - elsif [FoundationTypeBasementConditioned].include? @foundation_type + when FoundationTypeBasementConditioned basement = XMLHelper.add_element(foundation_type_el, 'Basement') XMLHelper.add_element(basement, 'Conditioned', true, :boolean) - elsif [FoundationTypeBasementUnconditioned].include? @foundation_type + when FoundationTypeBasementUnconditioned basement = XMLHelper.add_element(foundation_type_el, 'Basement') XMLHelper.add_element(basement, 'Conditioned', false, :boolean) - elsif [FoundationTypeCrawlspaceVented].include? @foundation_type + when FoundationTypeCrawlspaceVented crawlspace = XMLHelper.add_element(foundation_type_el, 'Crawlspace') XMLHelper.add_element(crawlspace, 'Vented', true, :boolean) if not @vented_crawlspace_sla.nil? @@ -3737,13 +3764,13 @@ def to_doc(building) XMLHelper.add_element(ventilation_rate, 'UnitofMeasure', UnitsSLA, :string) XMLHelper.add_element(ventilation_rate, 'Value', @vented_crawlspace_sla, :float, @vented_crawlspace_sla_isdefaulted) end - elsif [FoundationTypeCrawlspaceUnvented].include? @foundation_type + when FoundationTypeCrawlspaceUnvented crawlspace = XMLHelper.add_element(foundation_type_el, 'Crawlspace') XMLHelper.add_element(crawlspace, 'Vented', false, :boolean) - elsif @foundation_type == FoundationTypeCrawlspaceConditioned + when FoundationTypeCrawlspaceConditioned crawlspace = XMLHelper.add_element(foundation_type_el, 'Crawlspace') XMLHelper.add_element(crawlspace, 'Conditioned', true, :boolean) - elsif @foundation_type == FoundationTypeBellyAndWing + when FoundationTypeBellyAndWing belly_and_wing = XMLHelper.add_element(foundation_type_el, 'BellyAndWing') XMLHelper.add_element(belly_and_wing, 'SkirtPresent', @belly_wing_skirt_present, :boolean, @belly_wing_skirt_present_isdefaulted) unless @belly_wing_skirt_present.nil? else @@ -7723,11 +7750,12 @@ def to_doc(building) hvac_distribution = XMLHelper.add_element(hvac, 'HVACDistribution') sys_id = XMLHelper.add_element(hvac_distribution, 'SystemIdentifier') XMLHelper.add_attribute(sys_id, 'id', @id) - distribution_system_type_el = XMLHelper.add_element(hvac_distribution, 'DistributionSystemType') if [HVACDistributionTypeAir, HVACDistributionTypeHydronic].include? @distribution_system_type + distribution_system_type_el = XMLHelper.add_element(hvac_distribution, 'DistributionSystemType') XMLHelper.add_element(distribution_system_type_el, @distribution_system_type) XMLHelper.add_element(hvac_distribution, 'ConditionedFloorAreaServed', @conditioned_floor_area_served, :float) unless @conditioned_floor_area_served.nil? elsif [HVACDistributionTypeDSE].include? @distribution_system_type + distribution_system_type_el = XMLHelper.add_element(hvac_distribution, 'DistributionSystemType') XMLHelper.add_element(distribution_system_type_el, 'Other', @distribution_system_type, :string) XMLHelper.add_element(hvac_distribution, 'AnnualHeatingDistributionSystemEfficiency', @annual_heating_dse, :float) unless @annual_heating_dse.nil? XMLHelper.add_element(hvac_distribution, 'AnnualCoolingDistributionSystemEfficiency', @annual_cooling_dse, :float) unless @annual_cooling_dse.nil? diff --git a/HPXMLtoOpenStudio/resources/hpxml_schematron/EPvalidator.xml b/HPXMLtoOpenStudio/resources/hpxml_schematron/EPvalidator.xml index 0379fec462..f7db0334d9 100644 --- a/HPXMLtoOpenStudio/resources/hpxml_schematron/EPvalidator.xml +++ b/HPXMLtoOpenStudio/resources/hpxml_schematron/EPvalidator.xml @@ -18,6 +18,7 @@ Expected 0 or more element(s) for xpath: extension/EmissionsScenarios/EmissionsScenario Expected 0 or more element(s) for xpath: extension/UtilityBillScenarios/UtilityBillScenario Expected 0 or more element(s) for xpath: extension/UnavailablePeriods/UnavailablePeriod + Expected 0 or 1 element(s) for xpath: extension/WholeSFAorMFBuildingSimulation extension/SchedulesFilePath has been replaced by /HPXML/Building/BuildingDetails/BuildingSummary/extension/SchedulesFilePath extension/HVACSizingControl has been replaced by /HPXML/Building/BuildingDetails/BuildingSummary/extension/HVACSizingControl diff --git a/HPXMLtoOpenStudio/resources/hvac.rb b/HPXMLtoOpenStudio/resources/hvac.rb index 6775d0a7be..6e91052f60 100644 --- a/HPXMLtoOpenStudio/resources/hvac.rb +++ b/HPXMLtoOpenStudio/resources/hvac.rb @@ -10,7 +10,7 @@ module HVAC AirSourceCoolRatedIWB = 67.0 # degF, Rated indoor wetbulb for air-source systems, cooling CrankcaseHeaterTemp = 50.0 # degF - # TODO + # Adds any HVAC Systems to the OpenStudio model. # # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings # @param model [OpenStudio::Model::Model] OpenStudio Model object @@ -89,16 +89,12 @@ def self.apply_cooling_system(runner, model, weather, spaces, hpxml_bldg, hpxml_ end sys_id = cooling_system.id - if [HPXML::HVACTypeCentralAirConditioner, - HPXML::HVACTypeRoomAirConditioner, - HPXML::HVACTypeMiniSplitAirConditioner, - HPXML::HVACTypePTAC].include? cooling_system.cooling_system_type - + case cooling_system.cooling_system_type + when HPXML::HVACTypeCentralAirConditioner, HPXML::HVACTypeRoomAirConditioner, + HPXML::HVACTypeMiniSplitAirConditioner, HPXML::HVACTypePTAC airloop_map[sys_id] = apply_air_source_hvac_systems(model, runner, weather, cooling_system, heating_system, hvac_sequential_load_fracs, conditioned_zone, hvac_unavailable_periods, schedules_file, hpxml_bldg, hpxml_header) - - elsif [HPXML::HVACTypeEvaporativeCooler].include? cooling_system.cooling_system_type - + when HPXML::HVACTypeEvaporativeCooler airloop_map[sys_id] = apply_evaporative_cooler(model, cooling_system, hvac_sequential_load_fracs, conditioned_zone, hvac_unavailable_periods, hpxml_bldg.building_construction.number_of_units) end @@ -154,25 +150,16 @@ def self.apply_heating_system(runner, model, weather, spaces, hpxml_bldg, hpxml_ end sys_id = heating_system.id - if [HPXML::HVACTypeFurnace].include? heating_system.heating_system_type - + case heating_system.heating_system_type + when HPXML::HVACTypeFurnace airloop_map[sys_id] = apply_air_source_hvac_systems(model, runner, weather, nil, heating_system, hvac_sequential_load_fracs, conditioned_zone, hvac_unavailable_periods, schedules_file, hpxml_bldg, hpxml_header) - - elsif [HPXML::HVACTypeBoiler].include? heating_system.heating_system_type - + when HPXML::HVACTypeBoiler airloop_map[sys_id] = apply_boiler(model, runner, heating_system, hvac_sequential_load_fracs, conditioned_zone, hvac_unavailable_periods) - - elsif [HPXML::HVACTypeElectricResistance].include? heating_system.heating_system_type - + when HPXML::HVACTypeElectricResistance apply_electric_baseboard(model, heating_system, hvac_sequential_load_fracs, conditioned_zone, hvac_unavailable_periods) - - elsif [HPXML::HVACTypeStove, - HPXML::HVACTypeSpaceHeater, - HPXML::HVACTypeWallFurnace, - HPXML::HVACTypeFloorFurnace, - HPXML::HVACTypeFireplace].include? heating_system.heating_system_type - + when HPXML::HVACTypeStove, HPXML::HVACTypeSpaceHeater, HPXML::HVACTypeWallFurnace, + HPXML::HVACTypeFloorFurnace, HPXML::HVACTypeFireplace apply_unit_heater(model, heating_system, hvac_sequential_load_fracs, conditioned_zone, hvac_unavailable_periods) end @@ -223,24 +210,17 @@ def self.apply_heat_pump(runner, model, weather, spaces, hpxml_bldg, hpxml_heade hvac_remaining_load_fracs[:clg] -= heat_pump.fraction_cool_load_served sys_id = heat_pump.id - if [HPXML::HVACTypeHeatPumpWaterLoopToAir].include? heat_pump.heat_pump_type - + case heat_pump.heat_pump_type + when HPXML::HVACTypeHeatPumpWaterLoopToAir airloop_map[sys_id] = apply_water_loop_to_air_heat_pump(model, heat_pump, hvac_sequential_load_fracs, conditioned_zone, hvac_unavailable_periods) - - elsif [HPXML::HVACTypeHeatPumpAirToAir, - HPXML::HVACTypeHeatPumpMiniSplit, - HPXML::HVACTypeHeatPumpPTHP, - HPXML::HVACTypeHeatPumpRoom].include? heat_pump.heat_pump_type - + when HPXML::HVACTypeHeatPumpAirToAir, HPXML::HVACTypeHeatPumpMiniSplit, + HPXML::HVACTypeHeatPumpPTHP, HPXML::HVACTypeHeatPumpRoom airloop_map[sys_id] = apply_air_source_hvac_systems(model, runner, weather, heat_pump, heat_pump, hvac_sequential_load_fracs, conditioned_zone, hvac_unavailable_periods, schedules_file, hpxml_bldg, hpxml_header) - - elsif [HPXML::HVACTypeHeatPumpGroundToAir].include? heat_pump.heat_pump_type - + when HPXML::HVACTypeHeatPumpGroundToAir airloop_map[sys_id] = apply_ground_to_air_heat_pump(model, runner, weather, heat_pump, hvac_sequential_load_fracs, conditioned_zone, hpxml_bldg.site.ground_conductivity, hpxml_bldg.site.ground_diffusivity, hvac_unavailable_periods, hpxml_bldg.building_construction.number_of_units) - end next if heat_pump.backup_system.nil? @@ -291,21 +271,23 @@ def self.apply_air_source_hvac_systems(model, runner, weather, cooling_system, h if not cooling_system.nil? if cooling_system.is_a? HPXML::HeatPump is_heatpump = true - if cooling_system.heat_pump_type == HPXML::HVACTypeHeatPumpAirToAir + case cooling_system.heat_pump_type + when HPXML::HVACTypeHeatPumpAirToAir obj_name = Constants::ObjectTypeAirSourceHeatPump - elsif cooling_system.heat_pump_type == HPXML::HVACTypeHeatPumpMiniSplit + when HPXML::HVACTypeHeatPumpMiniSplit obj_name = Constants::ObjectTypeMiniSplitHeatPump - elsif cooling_system.heat_pump_type == HPXML::HVACTypeHeatPumpPTHP + when HPXML::HVACTypeHeatPumpPTHP obj_name = Constants::ObjectTypePTHP fan_watts_per_cfm = 0.0 - elsif cooling_system.heat_pump_type == HPXML::HVACTypeHeatPumpRoom + when HPXML::HVACTypeHeatPumpRoom obj_name = Constants::ObjectTypeRoomHP fan_watts_per_cfm = 0.0 else fail "Unexpected heat pump type: #{cooling_system.heat_pump_type}." end elsif cooling_system.is_a? HPXML::CoolingSystem - if cooling_system.cooling_system_type == HPXML::HVACTypeCentralAirConditioner + case cooling_system.cooling_system_type + when HPXML::HVACTypeCentralAirConditioner if heating_system.nil? obj_name = Constants::ObjectTypeCentralAirConditioner else @@ -315,14 +297,14 @@ def self.apply_air_source_hvac_systems(model, runner, weather, cooling_system, h fail "Fan powers for heating system '#{heating_system.id}' and cooling system '#{cooling_system.id}' are attached to a single distribution system and therefore must be the same." end end - elsif [HPXML::HVACTypeRoomAirConditioner, HPXML::HVACTypePTAC].include? cooling_system.cooling_system_type + when HPXML::HVACTypeRoomAirConditioner, HPXML::HVACTypePTAC fan_watts_per_cfm = 0.0 if cooling_system.cooling_system_type == HPXML::HVACTypeRoomAirConditioner obj_name = Constants::ObjectTypeRoomAC else obj_name = Constants::ObjectTypePTAC end - elsif cooling_system.cooling_system_type == HPXML::HVACTypeMiniSplitAirConditioner + when HPXML::HVACTypeMiniSplitAirConditioner obj_name = Constants::ObjectTypeMiniSplitAirConditioner else fail "Unexpected cooling system type: #{cooling_system.cooling_system_type}." @@ -681,20 +663,19 @@ def self.apply_ground_to_air_heat_pump(model, runner, weather, heat_pump, hvac_s xing.setAverageSoilSurfaceTemperature(ground_heat_exch_vert.groundTemperature.get) # Plant Loop - plant_loop = OpenStudio::Model::PlantLoop.new(model) - plant_loop.setName(obj_name + ' condenser loop') - plant_loop.setFluidType(hp_ap.fluid_type) - if hp_ap.fluid_type != EPlus::FluidWater - plant_loop.setGlycolConcentration((hp_ap.frac_glycol * 100).to_i) - end - plant_loop.setMaximumLoopTemperature(48.88889) - plant_loop.setMinimumLoopTemperature(UnitConversions.convert(hp_ap.design_hw, 'F', 'C')) - plant_loop.setMinimumLoopFlowRate(0) - plant_loop.setLoadDistributionScheme('SequentialLoad') + plant_loop = Model.add_plant_loop( + model, + name: "#{obj_name} condenser loop", + fluid_type: hp_ap.fluid_type, + glycol_concentration: (hp_ap.frac_glycol * 100).to_i, + min_temp: UnitConversions.convert(hp_ap.design_hw, 'F', 'C'), + max_temp: 48.88889, + max_flow_rate: UnitConversions.convert(geothermal_loop.loop_flow, 'gal/min', 'm^3/s') + ) + plant_loop.addSupplyBranchForComponent(ground_heat_exch_vert) plant_loop.addDemandBranchForComponent(htg_coil) plant_loop.addDemandBranchForComponent(clg_coil) - plant_loop.setMaximumLoopFlowRate(UnitConversions.convert(geothermal_loop.loop_flow, 'gal/min', 'm^3/s')) sizing_plant = plant_loop.sizingPlant sizing_plant.setLoopType('Condenser') @@ -745,7 +726,7 @@ def self.apply_ground_to_air_heat_pump(model, runner, weather, heat_pump, hvac_s add_pump_power_ems_program(model, pump_w, pump, air_loop_unitary) if heat_pump.is_shared_system - # Shared pump power per ANSI/RESNET/ICC 301-2019 Section 4.4.5.1 (pump runs 8760) + # Shared pump power per ANSI/RESNET/ICC 301-2022 Section 4.4.5.1 (pump runs 8760) design_level = heat_pump.shared_loop_watts / heat_pump.number_of_units_served.to_f equip = Model.add_electric_equipment( @@ -857,13 +838,10 @@ def self.apply_boiler(model, runner, heating_system, hvac_sequential_load_fracs, end # Plant Loop - plant_loop = OpenStudio::Model::PlantLoop.new(model) - plant_loop.setName(obj_name + ' hydronic heat loop') - plant_loop.setFluidType(EPlus::FluidWater) - plant_loop.setMaximumLoopTemperature(100) - plant_loop.setMinimumLoopTemperature(0) - plant_loop.setMinimumLoopFlowRate(0) - plant_loop.autocalculatePlantLoopVolume() + plant_loop = Model.add_plant_loop( + model, + name: "#{obj_name} hydronic heat loop" + ) loop_sizing = plant_loop.sizingPlant loop_sizing.setLoopType('Heating') @@ -1311,14 +1289,14 @@ def self.apply_ceiling_fans(runner, model, spaces, weather, hpxml_bldg, hpxml_he ceiling_fan = hpxml_bldg.ceiling_fans[0] obj_name = Constants::ObjectTypeCeilingFan - hrs_per_day = 10.5 # From ANSI 301-2019 + hrs_per_day = 10.5 # From ANSI/RESNET/ICC 301-2022 cfm_per_w = ceiling_fan.efficiency label_energy_use = ceiling_fan.label_energy_use count = ceiling_fan.count if !label_energy_use.nil? # priority if both provided annual_kwh = UnitConversions.convert(count * label_energy_use * hrs_per_day * 365.0, 'Wh', 'kWh') elsif !cfm_per_w.nil? - medium_cfm = 3000.0 # cfm, per ANSI 301-2019 + medium_cfm = 3000.0 # cfm, per ANSI/RESNET/ICC 301-2019 annual_kwh = UnitConversions.convert(count * medium_cfm / cfm_per_w * hrs_per_day * 365.0, 'Wh', 'kWh') end @@ -1593,10 +1571,11 @@ def self.get_cooling_setpoints(hvac_control, has_ceiling_fan, year, weather, off # @param compressor_type [TODO] TODO # @return [TODO] TODO def self.get_cool_cap_eir_ft_spec(compressor_type) - if compressor_type == HPXML::HVACCompressorTypeSingleStage + case compressor_type + when HPXML::HVACCompressorTypeSingleStage cap_ft_spec = [[3.68637657, -0.098352478, 0.000956357, 0.005838141, -0.0000127, -0.000131702]] eir_ft_spec = [[-3.437356399, 0.136656369, -0.001049231, -0.0079378, 0.000185435, -0.0001441]] - elsif compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage cap_ft_spec = [[3.998418659, -0.108728222, 0.001056818, 0.007512314, -0.0000139, -0.000164716], [3.466810106, -0.091476056, 0.000901205, 0.004163355, -0.00000919, -0.000110829]] eir_ft_spec = [[-4.282911381, 0.181023691, -0.001357391, -0.026310378, 0.000333282, -0.000197405], @@ -1610,17 +1589,18 @@ def self.get_cool_cap_eir_ft_spec(compressor_type) # @param compressor_type [TODO] TODO # @return [TODO] TODO def self.get_cool_cap_eir_fflow_spec(compressor_type) - if compressor_type == HPXML::HVACCompressorTypeSingleStage + case compressor_type + when HPXML::HVACCompressorTypeSingleStage # Single stage systems have PSC or constant torque ECM blowers, so the airflow rate is affected by the static pressure losses. cap_fflow_spec = [[0.718664047, 0.41797409, -0.136638137]] eir_fflow_spec = [[1.143487507, -0.13943972, -0.004047787]] - elsif compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage # Most two stage systems have PSC or constant torque ECM blowers, so the airflow rate is affected by the static pressure losses. cap_fflow_spec = [[0.655239515, 0.511655216, -0.166894731], [0.618281092, 0.569060264, -0.187341356]] eir_fflow_spec = [[1.639108268, -0.998953996, 0.359845728], [1.570774717, -0.914152018, 0.343377302]] - elsif compressor_type == HPXML::HVACCompressorTypeVariableSpeed + when HPXML::HVACCompressorTypeVariableSpeed # Variable speed systems have constant flow ECM blowers, so the air handler can always achieve the design airflow rate by sacrificing blower power. # So we assume that there is only one corresponding airflow rate for each compressor speed. eir_fflow_spec = [[1, 0, 0]] * 2 @@ -1655,17 +1635,18 @@ def self.get_heat_cap_eir_ft_spec(compressor_type, heating_capacity_retention_te # @param compressor_type [TODO] TODO # @return [TODO] TODO def self.get_heat_cap_eir_fflow_spec(compressor_type) - if compressor_type == HPXML::HVACCompressorTypeSingleStage + case compressor_type + when HPXML::HVACCompressorTypeSingleStage # Single stage systems have PSC or constant torque ECM blowers, so the airflow rate is affected by the static pressure losses. cap_fflow_spec = [[0.694045465, 0.474207981, -0.168253446]] eir_fflow_spec = [[2.185418751, -1.942827919, 0.757409168]] - elsif compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage # Most two stage systems have PSC or constant torque ECM blowers, so the airflow rate is affected by the static pressure losses. cap_fflow_spec = [[0.741466907, 0.378645444, -0.119754733], [0.76634609, 0.32840943, -0.094701495]] eir_fflow_spec = [[2.153618211, -1.737190609, 0.584269478], [2.001041353, -1.58869128, 0.587593517]] - elsif compressor_type == HPXML::HVACCompressorTypeVariableSpeed + when HPXML::HVACCompressorTypeVariableSpeed # Variable speed systems have constant flow ECM blowers, so the air handler can always achieve the design airflow rate by sacrificing blower power. # So we assume that there is only one corresponding airflow rate for each compressor speed. cap_fflow_spec = [[1, 0, 0]] * 3 @@ -1686,7 +1667,8 @@ def self.set_cool_curves_central_air_source(cooling_system, use_eer = false) set_cool_c_d(cooling_system) seer = cooling_system.cooling_efficiency_seer - if cooling_system.compressor_type == HPXML::HVACCompressorTypeSingleStage + case cooling_system.compressor_type + when HPXML::HVACCompressorTypeSingleStage clg_ap.cool_cap_ft_spec, clg_ap.cool_eir_ft_spec = get_cool_cap_eir_ft_spec(cooling_system.compressor_type) if not use_eer clg_ap.cool_rated_airflow_rate = clg_ap.cool_rated_cfm_per_ton[0] @@ -1699,7 +1681,7 @@ def self.set_cool_curves_central_air_source(cooling_system, use_eer = false) clg_ap.cool_eir_fflow_spec = [[1.0, 0.0, 0.0]] end - elsif cooling_system.compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage clg_ap.cool_rated_airflow_rate = clg_ap.cool_rated_cfm_per_ton[-1] clg_ap.cool_fan_speed_ratios = calc_fan_speed_ratios(clg_ap.cool_capacity_ratios, clg_ap.cool_rated_cfm_per_ton, clg_ap.cool_rated_airflow_rate) clg_ap.cool_cap_ft_spec, clg_ap.cool_eir_ft_spec = get_cool_cap_eir_ft_spec(cooling_system.compressor_type) @@ -1707,7 +1689,7 @@ def self.set_cool_curves_central_air_source(cooling_system, use_eer = false) clg_ap.cool_rated_cops = [0.2773 * seer - 0.0018] # Regression based on inverse model clg_ap.cool_rated_cops << clg_ap.cool_rated_cops[0] * 0.91 # COP ratio based on Dylan's data as seen in BEopt 2.8 options - elsif cooling_system.compressor_type == HPXML::HVACCompressorTypeVariableSpeed + when HPXML::HVACCompressorTypeVariableSpeed clg_ap.cooling_capacity_retention_temperature = 82.0 clg_ap.cooling_capacity_retention_fraction = 1.033 # From NEEP data clg_ap.cool_rated_airflow_rate = clg_ap.cool_rated_cfm_per_ton[-1] @@ -1724,11 +1706,12 @@ def self.set_cool_curves_central_air_source(cooling_system, use_eer = false) # @return [TODO] TODO def self.get_cool_capacity_ratios(hvac_system) # For each speed, ratio of capacity to nominal capacity - if hvac_system.compressor_type == HPXML::HVACCompressorTypeSingleStage + case hvac_system.compressor_type + when HPXML::HVACCompressorTypeSingleStage return [1.0] - elsif hvac_system.compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage return [0.72, 1.0] - elsif hvac_system.compressor_type == HPXML::HVACCompressorTypeVariableSpeed + when HPXML::HVACCompressorTypeVariableSpeed is_ducted = !hvac_system.distribution_system_idref.nil? if is_ducted return [0.394, 1.0] @@ -1753,7 +1736,8 @@ def self.set_heat_curves_central_air_source(heating_system, use_cop = false) set_heat_c_d(heating_system) hspf = heating_system.heating_efficiency_hspf - if heating_system.compressor_type == HPXML::HVACCompressorTypeSingleStage + case heating_system.compressor_type + when HPXML::HVACCompressorTypeSingleStage heating_capacity_retention_temp, heating_capacity_retention_fraction = get_heating_capacity_retention(heating_system) htg_ap.heat_cap_ft_spec, htg_ap.heat_eir_ft_spec = get_heat_cap_eir_ft_spec(heating_system.compressor_type, heating_capacity_retention_temp, heating_capacity_retention_fraction) if not use_cop @@ -1764,7 +1748,7 @@ def self.set_heat_curves_central_air_source(heating_system, use_cop = false) htg_ap.heat_fan_speed_ratios = [1.0] end - elsif heating_system.compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage heating_capacity_retention_temp, heating_capacity_retention_fraction = get_heating_capacity_retention(heating_system) htg_ap.heat_cap_ft_spec, htg_ap.heat_eir_ft_spec = get_heat_cap_eir_ft_spec(heating_system.compressor_type, heating_capacity_retention_temp, heating_capacity_retention_fraction) htg_ap.heat_rated_airflow_rate = htg_ap.heat_rated_cfm_per_ton[-1] @@ -1772,7 +1756,7 @@ def self.set_heat_curves_central_air_source(heating_system, use_cop = false) htg_ap.heat_rated_cops = [0.0426 * hspf**2 - 0.0747 * hspf + 1.5374] # Regression based on inverse model htg_ap.heat_rated_cops << htg_ap.heat_rated_cops[0] * 0.87 # COP ratio based on Dylan's data as seen in BEopt 2.8 options - elsif heating_system.compressor_type == HPXML::HVACCompressorTypeVariableSpeed + when HPXML::HVACCompressorTypeVariableSpeed htg_ap.heat_rated_airflow_rate = htg_ap.heat_rated_cfm_per_ton[-1] htg_ap.heat_capacity_ratios = get_heat_capacity_ratios(heating_system) htg_ap.heat_fan_speed_ratios = calc_fan_speed_ratios(htg_ap.heat_capacity_ratios, htg_ap.heat_rated_cfm_per_ton, htg_ap.heat_rated_airflow_rate) @@ -1889,11 +1873,12 @@ def self.set_cool_detailed_performance_data(heat_pump) # @return [TODO] TODO def self.get_heat_capacity_ratios(heat_pump) # For each speed, ratio of capacity to nominal capacity - if heat_pump.compressor_type == HPXML::HVACCompressorTypeSingleStage + case heat_pump.compressor_type + when HPXML::HVACCompressorTypeSingleStage return [1.0] - elsif heat_pump.compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage return [0.72, 1.0] - elsif heat_pump.compressor_type == HPXML::HVACCompressorTypeVariableSpeed + when HPXML::HVACCompressorTypeVariableSpeed is_ducted = !heat_pump.distribution_system_idref.nil? if is_ducted nominal_to_max_ratio = 0.972 @@ -1975,15 +1960,16 @@ def self.get_cool_cfm_per_ton(compressor_type, use_eer = false) # @return [TODO] TODO def self.get_heat_cfm_per_ton(compressor_type, use_cop_or_htg_sys = false) # cfm/ton of rated capacity - if compressor_type == HPXML::HVACCompressorTypeSingleStage + case compressor_type + when HPXML::HVACCompressorTypeSingleStage if not use_cop_or_htg_sys return [384.1] else return [350] end - elsif compressor_type == HPXML::HVACCompressorTypeTwoStage + when HPXML::HVACCompressorTypeTwoStage return [391.3333, 352.2] - elsif compressor_type == HPXML::HVACCompressorTypeVariableSpeed + when HPXML::HVACCompressorTypeVariableSpeed return [400.0, 400.0, 400.0] else fail 'Compressor type not supported.' @@ -4430,16 +4416,19 @@ def self.set_cool_c_d(cooling_system) if ((cooling_system.is_a? HPXML::CoolingSystem) && ([HPXML::HVACTypeRoomAirConditioner, HPXML::HVACTypePTAC].include? cooling_system.cooling_system_type)) || ((cooling_system.is_a? HPXML::HeatPump) && ([HPXML::HVACTypeHeatPumpPTHP, HPXML::HVACTypeHeatPumpRoom].include? cooling_system.heat_pump_type)) clg_ap.cool_c_d = 0.22 - elsif cooling_system.compressor_type == HPXML::HVACCompressorTypeSingleStage - if cooling_system.cooling_efficiency_seer < 13.0 - clg_ap.cool_c_d = 0.20 - else - clg_ap.cool_c_d = 0.07 + else + case cooling_system.compressor_type + when HPXML::HVACCompressorTypeSingleStage + if cooling_system.cooling_efficiency_seer < 13.0 + clg_ap.cool_c_d = 0.20 + else + clg_ap.cool_c_d = 0.07 + end + when HPXML::HVACCompressorTypeTwoStage + clg_ap.cool_c_d = 0.11 + when HPXML::HVACCompressorTypeVariableSpeed + clg_ap.cool_c_d = 0.25 end - elsif cooling_system.compressor_type == HPXML::HVACCompressorTypeTwoStage - clg_ap.cool_c_d = 0.11 - elsif cooling_system.compressor_type == HPXML::HVACCompressorTypeVariableSpeed - clg_ap.cool_c_d = 0.25 end # PLF curve @@ -4457,16 +4446,19 @@ def self.set_heat_c_d(heating_system) # Degradation coefficient for heating if (heating_system.is_a? HPXML::HeatPump) && ([HPXML::HVACTypeHeatPumpPTHP, HPXML::HVACTypeHeatPumpRoom].include? heating_system.heat_pump_type) htg_ap.heat_c_d = 0.22 - elsif heating_system.compressor_type == HPXML::HVACCompressorTypeSingleStage - if heating_system.heating_efficiency_hspf < 7.0 - htg_ap.heat_c_d = 0.20 - else + else + case heating_system.compressor_type + when HPXML::HVACCompressorTypeSingleStage + if heating_system.heating_efficiency_hspf < 7.0 + htg_ap.heat_c_d = 0.20 + else + htg_ap.heat_c_d = 0.11 + end + when HPXML::HVACCompressorTypeTwoStage htg_ap.heat_c_d = 0.11 + when HPXML::HVACCompressorTypeVariableSpeed + htg_ap.heat_c_d = 0.25 end - elsif heating_system.compressor_type == HPXML::HVACCompressorTypeTwoStage - htg_ap.heat_c_d = 0.11 - elsif heating_system.compressor_type == HPXML::HVACCompressorTypeVariableSpeed - htg_ap.heat_c_d = 0.25 end # PLF curve @@ -4544,14 +4536,15 @@ def self.set_gshp_assumptions(heat_pump, weather) end pipe_diameter = geothermal_loop.pipe_diameter # Pipe nominal size conversion to pipe outside diameter and inside diameter, - # only pipe sizes <= 2" are used here with DR11 (dimension ratio), - if pipe_diameter == 0.75 # 3/4" pipe + # only pipe sizes <= 2" are used here with DR11 (dimension ratio) + case pipe_diameter + when 0.75 # 3/4" pipe hp_ap.pipe_od = 1.050 # in hp_ap.pipe_id = 0.859 # in - elsif pipe_diameter == 1.0 # 1" pipe + when 1.0 # 1" pipe hp_ap.pipe_od = 1.315 # in hp_ap.pipe_id = 1.076 # in - elsif pipe_diameter == 1.25 # 1-1/4" pipe + when 1.25 # 1-1/4" pipe hp_ap.pipe_od = 1.660 # in hp_ap.pipe_id = 1.358 # in else @@ -4559,13 +4552,14 @@ def self.set_gshp_assumptions(heat_pump, weather) end hp_ap.u_tube_spacing_type = 'b' # Calculate distance between pipes - if hp_ap.u_tube_spacing_type == 'as' + case hp_ap.u_tube_spacing_type + when 'as' # Two tubes, spaced 1/8ā€ apart at the center of the borehole hp_ap.u_tube_spacing = 0.125 - elsif hp_ap.u_tube_spacing_type == 'b' + when 'b' # Two tubes equally spaced between the borehole edges hp_ap.u_tube_spacing = 0.9661 - elsif hp_ap.u_tube_spacing_type == 'c' + when 'c' # Both tubes placed against outer edge of borehole hp_ap.u_tube_spacing = geothermal_loop.bore_diameter - 2 * hp_ap.pipe_od end @@ -5288,7 +5282,7 @@ def self.apply_shared_cooling_systems(hpxml_bldg) aux_dweq = cooling_system.fan_coil_watts end end - # ANSI/RESNET/ICC 301-2019 Equation 4.4-2 + # ANSI/RESNET/ICC 301-2022 Equation 4.4-2 seer_eq = (cap - 3.41 * aux - 3.41 * aux_dweq * n_dweq) / (chiller_input + aux + aux_dweq * n_dweq) elsif cooling_system.cooling_system_type == HPXML::HVACTypeCoolingTower @@ -5301,7 +5295,7 @@ def self.apply_shared_cooling_systems(hpxml_bldg) wlhp_input = wlhp_cap / wlhp.cooling_efficiency_eer end end - # ANSI/RESNET/ICC 301-2019 Equation 4.4-3 + # ANSI/RESNET/ICC 301-2022 Equation 4.4-3 seer_eq = (wlhp_cap - 3.41 * aux / n_dweq) / (wlhp_input + aux / n_dweq) else @@ -5394,7 +5388,7 @@ def self.apply_shared_heating_systems(hpxml_bldg) if heating_system.heating_system_type == HPXML::HVACTypeBoiler && hydronic_type.to_s == HPXML::HydronicTypeWaterLoop # Shared boiler w/ water loop heat pump - # Per ANSI/RESNET/ICC 301-2019 Section 4.4.7.2, model as: + # Per ANSI/RESNET/ICC 301-2022 Section 4.4.7.2, model as: # A) heat pump with constant efficiency and duct losses, fraction heat load served = 1/COP # B) boiler, fraction heat load served = 1-1/COP fraction_heat_load_served = heating_system.fraction_heat_load_served @@ -5586,7 +5580,7 @@ def self.apply_unit_multiplier(hpxml_bldg, hpxml_header) # Calculates rated SEER (older metric) from rated SEER2 (newer metric). # # Source: ANSI/RESNET/ICC 301 Table 4.4.4.1(1) SEER2/HSPF2 Conversion Factors - # This is based on a regression of products, not a translation. + # Note that this is a regression based on products on the market, not a conversion. # # @param seer2 [Double] Cooling efficiency (Btu/Wh) # @param is_ducted [Boolean] True if a ducted HVAC system diff --git a/HPXMLtoOpenStudio/resources/hvac_sizing.rb b/HPXMLtoOpenStudio/resources/hvac_sizing.rb index 408e380b02..400c639997 100644 --- a/HPXMLtoOpenStudio/resources/hvac_sizing.rb +++ b/HPXMLtoOpenStudio/resources/hvac_sizing.rb @@ -260,10 +260,11 @@ def self.process_site_calcs_and_design_temps(mj, weather, hpxml_bldg) locations.uniq.each do |location| next if [HPXML::LocationGround].include? location - if [HPXML::LocationOutside, HPXML::LocationRoofDeck, HPXML::LocationManufacturedHomeUnderBelly].include? location + case location + when HPXML::LocationOutside, HPXML::LocationRoofDeck, HPXML::LocationManufacturedHomeUnderBelly mj.cool_design_temps[location] = hpxml_bldg.header.manualj_cooling_design_temp mj.heat_design_temps[location] = hpxml_bldg.header.manualj_heating_design_temp - elsif HPXML::conditioned_locations.include? location + when *HPXML::conditioned_locations mj.cool_design_temps[location] = get_design_temp_cooling(mj, weather, HPXML::LocationConditionedSpace, hpxml_bldg) mj.heat_design_temps[location] = get_design_temp_heating(mj, weather, HPXML::LocationConditionedSpace, hpxml_bldg) else @@ -423,54 +424,62 @@ def self.get_design_temp_cooling(mj, weather, location, hpxml_bldg) end else if not roof.radiant_barrier - if roof.roof_type == HPXML::RoofTypeAsphaltShingles - if [HPXML::ColorDark, HPXML::ColorMediumDark].include? roof.roof_color + case roof.roof_type + when HPXML::RoofTypeAsphaltShingles + case roof.roof_color + when HPXML::ColorDark, HPXML::ColorMediumDark cool_temp += 130.0 * roof.net_area else cool_temp += 120.0 * roof.net_area end - elsif roof.roof_type == HPXML::RoofTypeWoodShingles + when HPXML::RoofTypeWoodShingles cool_temp += 120.0 * roof.net_area - elsif roof.roof_type == HPXML::RoofTypeMetal - if [HPXML::ColorDark, HPXML::ColorMediumDark].include? roof.roof_color + when HPXML::RoofTypeMetal + case roof.roof_color + when HPXML::ColorDark, HPXML::ColorMediumDark cool_temp += 130.0 * roof.net_area - elsif [HPXML::ColorMedium, HPXML::ColorLight].include? roof.roof_color + when HPXML::ColorMedium, HPXML::ColorLight cool_temp += 120.0 * roof.net_area - elsif [HPXML::ColorReflective].include? roof.roof_color + when HPXML::ColorReflective cool_temp += 95.0 * roof.net_area end - elsif roof.roof_type == HPXML::RoofTypeClayTile - if [HPXML::ColorDark, HPXML::ColorMediumDark].include? roof.roof_color + when HPXML::RoofTypeClayTile + case roof.roof_color + when HPXML::ColorDark, HPXML::ColorMediumDark cool_temp += 110.0 * roof.net_area - elsif [HPXML::ColorMedium, HPXML::ColorLight].include? roof.roof_color + when HPXML::ColorMedium, HPXML::ColorLight cool_temp += 105.0 * roof.net_area - elsif [HPXML::ColorReflective].include? roof.roof_color + when HPXML::ColorReflective cool_temp += 95.0 * roof.net_area end end else # with a radiant barrier - if roof.roof_type == HPXML::RoofTypeAsphaltShingles - if [HPXML::ColorDark, HPXML::ColorMediumDark].include? roof.roof_color + case roof.roof_type + when HPXML::RoofTypeAsphaltShingles + case roof.roof_color + when HPXML::ColorDark, HPXML::ColorMediumDark cool_temp += 120.0 * roof.net_area else cool_temp += 110.0 * roof.net_area end - elsif roof.roof_type == HPXML::RoofTypeWoodShingles + when HPXML::RoofTypeWoodShingles cool_temp += 110.0 * roof.net_area - elsif roof.roof_type == HPXML::RoofTypeMetal - if [HPXML::ColorDark, HPXML::ColorMediumDark].include? roof.roof_color + when HPXML::RoofTypeMetal + case roof.roof_color + when HPXML::ColorDark, HPXML::ColorMediumDark cool_temp += 120.0 * roof.net_area - elsif [HPXML::ColorMedium, HPXML::ColorLight].include? roof.roof_color + when HPXML::ColorMedium, HPXML::ColorLight cool_temp += 110.0 * roof.net_area - elsif [HPXML::ColorReflective].include? roof.roof_color + when HPXML::ColorReflective cool_temp += 95.0 * roof.net_area end - elsif roof.roof_type == HPXML::RoofTypeClayTile - if [HPXML::ColorDark, HPXML::ColorMediumDark].include? roof.roof_color + when HPXML::RoofTypeClayTile + case roof.roof_color + when HPXML::ColorDark, HPXML::ColorMediumDark cool_temp += 105.0 * roof.net_area - elsif [HPXML::ColorMedium, HPXML::ColorLight].include? roof.roof_color + when HPXML::ColorMedium, HPXML::ColorLight cool_temp += 100.0 * roof.net_area - elsif [HPXML::ColorReflective].include? roof.roof_color + when HPXML::ColorReflective cool_temp += 95.0 * roof.net_area end end @@ -1070,11 +1079,12 @@ def self.process_load_walls(mj, hpxml_bldg, all_zone_loads, all_space_loads) color = HPXML::ColorDark end - if color == HPXML::ColorLight + case color + when HPXML::ColorLight color_multiplier = 0.65 # MJ8 Table 4B Notes, pg 348 - elsif color == HPXML::ColorMedium + when HPXML::ColorMedium color_multiplier = 0.83 # MJ8 Appendix 12, pg 519 - elsif color == HPXML::ColorDark + when HPXML::ColorDark color_multiplier = 1.0 end @@ -1170,17 +1180,18 @@ def self.process_load_roofs(mj, hpxml_bldg, all_zone_loads, all_space_loads) end # Base CLTD color adjustment based on notes in MJ8 Figure A12-16 - if [HPXML::ColorDark, HPXML::ColorMediumDark].include? roof.roof_color + case roof.roof_color + when HPXML::ColorDark, HPXML::ColorMediumDark if [HPXML::RoofTypeClayTile, HPXML::RoofTypeWoodShingles].include? roof.roof_type cltd *= 0.83 end - elsif [HPXML::ColorMedium, HPXML::ColorLight].include? roof.roof_color + when HPXML::ColorMedium, HPXML::ColorLight if [HPXML::RoofTypeClayTile].include? roof.roof_type cltd *= 0.65 else cltd *= 0.83 end - elsif [HPXML::ColorReflective].include? roof.roof_color + when HPXML::ColorReflective if [HPXML::RoofTypeAsphaltShingles, HPXML::RoofTypeWoodShingles].include? roof.roof_type cltd *= 0.83 else @@ -1264,17 +1275,17 @@ def self.process_load_floors(mj, hpxml_bldg, all_zone_loads, all_space_loads) zone = space.zone has_radiant_floor = get_has_radiant_floor(zone) + u_floor = 1.0 / floor.insulation_assembly_r_value if floor.is_exterior htd_adj = mj.htd htd_adj += 25.0 if has_radiant_floor # Table 4A: Radiant floor over open crawlspace: HTM = U-Value Ɨ (HTD + 25) - clg_htm = (1.0 / floor.insulation_assembly_r_value) * (mj.ctd - 5.0 + mj.daily_range_temp_adjust[mj.daily_range_num]) - htg_htm = (1.0 / floor.insulation_assembly_r_value) * htd_adj + clg_htm = u_floor * (mj.ctd - 5.0 + mj.daily_range_temp_adjust[mj.daily_range_num]) + htg_htm = u_floor * htd_adj else # Partition floor adjacent_space = floor.exterior_adjacent_to - if floor.is_floor && [HPXML::LocationCrawlspaceVented, HPXML::LocationCrawlspaceUnvented, HPXML::LocationBasementUnconditioned].include?(adjacent_space) - u_floor = 1.0 / floor.insulation_assembly_r_value + if [HPXML::LocationCrawlspaceVented, HPXML::LocationCrawlspaceUnvented, HPXML::LocationBasementUnconditioned].include?(adjacent_space) sum_ua_wall = 0.0 sum_a_wall = 0.0 @@ -1306,7 +1317,7 @@ def self.process_load_floors(mj, hpxml_bldg, all_zone_loads, all_space_loads) u_wall = sum_ua_wall / sum_a_wall htd_adj = mj.htd - htd_adj += 25.0 if has_radiant_floor && HPXML::LocationCrawlspaceVented # Table 4A: Radiant floor over open crawlspace: HTM = U-Value Ɨ (HTD + 25) + htd_adj += 25.0 if has_radiant_floor # Manual J Figure A12-6 footnote 2) # Calculate partition temperature different cooling (PTDC) per Manual J Figure A12-17 # Calculate partition temperature different heating (PTDH) per Manual J Figure A12-6 @@ -1320,19 +1331,21 @@ def self.process_load_floors(mj, hpxml_bldg, all_zone_loads, all_space_loads) ptdh_floor = u_wall * htd_adj / (4.0 * u_floor + u_wall) end - clg_htm = (1.0 / floor.insulation_assembly_r_value) * ptdc_floor - htg_htm = (1.0 / floor.insulation_assembly_r_value) * ptdh_floor + clg_htm = u_floor * ptdc_floor + htg_htm = u_floor * ptdh_floor else # E.g., floor over garage - clg_htm = (1.0 / floor.insulation_assembly_r_value) * (mj.cool_design_temps[adjacent_space] - mj.cool_setpoint) - htg_htm = (1.0 / floor.insulation_assembly_r_value) * (mj.heat_setpoint - mj.heat_design_temps[adjacent_space]) + htd_adj = mj.heat_setpoint - mj.heat_design_temps[adjacent_space] + htd_adj += 25.0 if has_radiant_floor # Manual J Figure A12-6 footnote 2), and Table 4A: Radiant floor over garage: HTM = U-Value Ɨ (HTD + 25) + clg_htm = u_floor * (mj.cool_design_temps[adjacent_space] - mj.cool_setpoint) + htg_htm = u_floor * htd_adj end end clg_loads = clg_htm * floor.net_area htg_loads = htg_htm * floor.net_area all_zone_loads[zone].Cool_Floors += clg_loads all_zone_loads[zone].Heat_Floors += htg_loads - all_space_loads[space].Cool_Roofs += clg_loads - all_space_loads[space].Heat_Roofs += htg_loads + all_space_loads[space].Cool_Floors += clg_loads + all_space_loads[space].Heat_Floors += htg_loads detailed_output_values = DetailedOutputValues.new(area: floor.net_area, heat_htm: htg_htm, cool_htm: clg_htm, @@ -1748,16 +1761,16 @@ def self.apply_hvac_loads_to_hvac_sizings(hvac_sizings, hvac_loads) def self.get_duct_regain_factor(duct, hpxml_bldg) f_regain = nil - if [HPXML::LocationOutside, HPXML::LocationRoofDeck].include? duct.duct_location + case duct.duct_location + when HPXML::LocationOutside, HPXML::LocationRoofDeck f_regain = 0.0 - elsif [HPXML::LocationOtherHousingUnit, HPXML::LocationOtherHeatedSpace, HPXML::LocationOtherMultifamilyBufferSpace, - HPXML::LocationOtherNonFreezingSpace, HPXML::LocationExteriorWall, HPXML::LocationUnderSlab, - HPXML::LocationManufacturedHomeBelly].include? duct.duct_location + when HPXML::LocationOtherHousingUnit, HPXML::LocationOtherHeatedSpace, HPXML::LocationOtherMultifamilyBufferSpace, + HPXML::LocationOtherNonFreezingSpace, HPXML::LocationExteriorWall, HPXML::LocationUnderSlab, HPXML::LocationManufacturedHomeBelly space_values = Geometry.get_temperature_scheduled_space_values(duct.duct_location) f_regain = space_values[:f_regain] - elsif [HPXML::LocationBasementUnconditioned, HPXML::LocationCrawlspaceVented, HPXML::LocationCrawlspaceUnvented].include? duct.duct_location + when HPXML::LocationBasementUnconditioned, HPXML::LocationCrawlspaceVented, HPXML::LocationCrawlspaceUnvented ceilings = hpxml_bldg.floors.select { |f| f.is_floor && [f.interior_adjacent_to, f.exterior_adjacent_to].include?(duct.duct_location) } avg_ceiling_rvalue = calculate_average_r_value(ceilings) @@ -1799,13 +1812,13 @@ def self.get_duct_regain_factor(duct, hpxml_bldg) end end - elsif [HPXML::LocationAtticVented, HPXML::LocationAtticUnvented].include? duct.duct_location + when HPXML::LocationAtticVented, HPXML::LocationAtticUnvented f_regain = 0.10 # This would likely be higher for unvented attics with roof insulation - elsif [HPXML::LocationGarage].include? duct.duct_location + when HPXML::LocationGarage f_regain = 0.05 - elsif HPXML::conditioned_locations.include? duct.duct_location + when *HPXML::conditioned_locations f_regain = 1.0 end @@ -1978,11 +1991,11 @@ def self.apply_hvac_duct_loads(mj, zone, hvac_loads, zone_loads, all_space_loads end end - # TODO + # Calculates the duct loads using Manual J Table 7 default duct tables. # # @param hpxml_bldg [HPXML::Building] HPXML Building object representing an individual dwelling unit # @param manualj_duct_load [HPXML::ManualJDuctLoad] Manual J duct load of interest - # @return [TODO] TODO + # @return [Array] Heating loss factor, Cooling sensible gain factor, Cooling latent gain Btuh def self.get_duct_table7_factors(hpxml_bldg, manualj_duct_load) # Gather values htg_oat = hpxml_bldg.header.manualj_heating_design_temp @@ -3422,13 +3435,14 @@ def self.apply_hvac_ground_loop(mj, runner, hvac_sizings, weather, hvac_cooling, def self.get_geothermal_loop_borefield_ft_per_ton(mj, hpxml_bldg, geothermal_loop, weather, hvac_cooling) hvac_cooling_ap = hvac_cooling.additional_properties - if hvac_cooling_ap.u_tube_spacing_type == 'b' + case hvac_cooling_ap.u_tube_spacing_type + when 'b' beta_0 = 17.4427 beta_1 = -0.6052 - elsif hvac_cooling_ap.u_tube_spacing_type == 'c' + when 'c' beta_0 = 21.9059 beta_1 = -0.3796 - elsif hvac_cooling_ap.u_tube_spacing_type == 'as' + when 'as' beta_0 = 20.1004 beta_1 = -0.94467 end @@ -4358,13 +4372,14 @@ def self.calculate_space_design_temp(mj, location, weather, hpxml_bldg, setpoint sum_uat, sum_ua = 0.0, 0.0 space_UAs.each do |ua_type, ua| - if ua_type == HPXML::LocationGround + case ua_type + when HPXML::LocationGround sum_uat += ua * ground_db sum_ua += ua - elsif ua_type == HPXML::LocationOutside + when HPXML::LocationOutside sum_uat += ua * design_db sum_ua += ua - elsif ua_type == HPXML::LocationConditionedSpace + when HPXML::LocationConditionedSpace sum_uat += ua * setpoint_temp sum_ua += ua else @@ -4393,11 +4408,12 @@ def self.calculate_space_design_temp(mj, location, weather, hpxml_bldg, setpoint ua_conditioned = 0.0 ua_outside = 0.0 space_UAs.each do |ua_type, ua| - if ua_type == HPXML::LocationOutside + case ua_type + when HPXML::LocationOutside ua_outside += ua - elsif ua_type == HPXML::LocationConditionedSpace + when HPXML::LocationConditionedSpace ua_conditioned += ua - elsif ua_type != HPXML::LocationGround + when !HPXML::LocationGround fail "Unexpected space ua type: '#{ua_type}'." end end @@ -5006,35 +5022,36 @@ def self.get_window_interior_shading_coefficient(window) end else # Use physical window properties - if window.glass_layers == HPXML::WindowLayersSinglePane - if [HPXML::WindowGlassTypeTintedReflective, - HPXML::WindowGlassTypeReflective].include? window.glass_type + case window.glass_layers + when HPXML::WindowLayersSinglePane + case window.glass_type + when HPXML::WindowGlassTypeTintedReflective, HPXML::WindowGlassTypeReflective window_type = '1P Reflective' - elsif [HPXML::WindowGlassTypeTinted].include? window.glass_type + when HPXML::WindowGlassTypeTinted window_type = '1P Heat Absorbing' else window_type = '1P Clear' end - elsif window.glass_layers == HPXML::WindowLayersDoublePane - if [HPXML::WindowGlassTypeTintedReflective, - HPXML::WindowGlassTypeReflective].include? window.glass_type + when HPXML::WindowLayersDoublePane + case window.glass_type + when HPXML::WindowGlassTypeTintedReflective, HPXML::WindowGlassTypeReflective window_type = '2P Reflective' - elsif [HPXML::WindowGlassTypeTinted].include? window.glass_type + when HPXML::WindowGlassTypeTinted window_type = '2P Heat Absorbing' - elsif [HPXML::WindowGlassTypeLowELowSolarGain].include? window.glass_type + when HPXML::WindowGlassTypeLowELowSolarGain window_type = '2P Low-e Option 3' - elsif [HPXML::WindowGlassTypeLowE].include? window.glass_type + when HPXML::WindowGlassTypeLowE window_type = '2P Low-e Option 2' - elsif [HPXML::WindowGlassTypeLowEHighSolarGain].include? window.glass_type + when HPXML::WindowGlassTypeLowEHighSolarGain window_type = '2P Low-e Option 1' else window_type = '2P Clear' end - elsif window.glass_layers == HPXML::WindowLayersTriplePane - if [HPXML::WindowGlassTypeTintedReflective, - HPXML::WindowGlassTypeReflective].include? window.glass_type + when HPXML::WindowLayersTriplePane + case window.glass_type + when HPXML::WindowGlassTypeTintedReflective, HPXML::WindowGlassTypeReflective window_type = '3P Reflective' - elsif [HPXML::WindowGlassTypeTinted].include? window.glass_type + when HPXML::WindowGlassTypeTinted window_type = '3P Heat Absorbing' else window_type = '3P Clear' diff --git a/HPXMLtoOpenStudio/resources/internal_gains.rb b/HPXMLtoOpenStudio/resources/internal_gains.rb index 3480915acd..281f2c0b81 100644 --- a/HPXMLtoOpenStudio/resources/internal_gains.rb +++ b/HPXMLtoOpenStudio/resources/internal_gains.rb @@ -13,11 +13,11 @@ module InternalGains # @return [nil] def self.apply_building_occupants(runner, model, hpxml_bldg, hpxml_header, spaces, schedules_file) if hpxml_bldg.building_occupancy.number_of_residents.nil? # Asset calculation - num_occ = Geometry.get_occupancy_default_num(nbeds: hpxml_bldg.building_construction.number_of_bedrooms) + n_occ = Geometry.get_occupancy_default_num(nbeds: hpxml_bldg.building_construction.number_of_bedrooms) else # Operational calculation - num_occ = hpxml_bldg.building_occupancy.number_of_residents + n_occ = hpxml_bldg.building_occupancy.number_of_residents end - return if num_occ <= 0 + return if n_occ <= 0 occ_gain, _hrs_per_day, sens_frac, _lat_frac = Defaults.get_occupancy_values() activity_per_person = UnitConversions.convert(occ_gain, 'Btu/hr', 'W') @@ -60,7 +60,7 @@ def self.apply_building_occupants(runner, model, hpxml_bldg, hpxml_header, space occ.setName(Constants::ObjectTypeOccupants) occ.setSpace(spaces[HPXML::LocationConditionedSpace]) occ_def.setName(Constants::ObjectTypeOccupants) - occ_def.setNumberofPeople(num_occ) + occ_def.setNumberofPeople(n_occ) occ_def.setFractionRadiant(occ_rad) occ_def.setSensibleHeatFraction(occ_sens) occ_def.setMeanRadiantTemperatureCalculationType('ZoneAveraged') @@ -81,11 +81,13 @@ def self.apply_building_occupants(runner, model, hpxml_bldg, hpxml_header, space # @param schedules_file [SchedulesFile] SchedulesFile wrapper class instance of detailed schedule files # @return [nil] def self.apply_general_water_use(runner, model, hpxml_bldg, hpxml_header, spaces, schedules_file) + nbeds = hpxml_bldg.building_construction.number_of_bedrooms + n_occ = hpxml_bldg.building_occupancy.number_of_residents + unit_type = hpxml_bldg.building_construction.residential_facility_type general_water_use_usage_multiplier = hpxml_bldg.building_occupancy.general_water_use_usage_multiplier - nbeds_eq = hpxml_bldg.building_construction.additional_properties.equivalent_number_of_bedrooms if not hpxml_header.apply_ashrae140_assumptions - water_sens_btu, water_lat_btu = Defaults.get_water_use_internal_gains(nbeds_eq, general_water_use_usage_multiplier) + water_sens_btu, water_lat_btu = Defaults.get_water_use_internal_gains(nbeds, n_occ, unit_type, general_water_use_usage_multiplier) # Create schedule water_schedule = nil diff --git a/HPXMLtoOpenStudio/resources/lighting.rb b/HPXMLtoOpenStudio/resources/lighting.rb index 53a0dfec1f..551f48acf4 100644 --- a/HPXMLtoOpenStudio/resources/lighting.rb +++ b/HPXMLtoOpenStudio/resources/lighting.rb @@ -17,6 +17,7 @@ def self.apply(runner, model, spaces, hpxml_bldg, hpxml_header, schedules_file) unit_multiplier = hpxml_bldg.building_construction.number_of_units cfa = hpxml_bldg.building_construction.conditioned_floor_area eri_version = hpxml_header.eri_calculation_version + n_occ = hpxml_bldg.building_occupancy.number_of_residents ltg_locns = [HPXML::LocationInterior, HPXML::LocationExterior, HPXML::LocationGarage] ltg_types = [HPXML::LightingTypeCFL, HPXML::LightingTypeLFL, HPXML::LightingTypeLED] @@ -39,7 +40,7 @@ def self.apply(runner, model, spaces, hpxml_bldg, hpxml_header, schedules_file) fractions[[HPXML::LocationInterior, HPXML::LightingTypeLFL]], fractions[[HPXML::LocationInterior, HPXML::LightingTypeLED]]) end - int_kwh = 0.0 if int_kwh.nil? + int_kwh = 0.0 if int_kwh.nil? || n_occ == 0 int_kwh *= lighting.interior_usage_multiplier unless lighting.interior_usage_multiplier.nil? # Calculate exterior lighting kWh/yr @@ -50,7 +51,7 @@ def self.apply(runner, model, spaces, hpxml_bldg, hpxml_header, schedules_file) fractions[[HPXML::LocationExterior, HPXML::LightingTypeLFL]], fractions[[HPXML::LocationExterior, HPXML::LightingTypeLED]]) end - ext_kwh = 0.0 if ext_kwh.nil? + ext_kwh = 0.0 if ext_kwh.nil? || n_occ == 0 ext_kwh *= lighting.exterior_usage_multiplier unless lighting.exterior_usage_multiplier.nil? ext_kwh *= unit_multiplier # Not in a thermal zone, so needs to be explicitly multiplied @@ -69,7 +70,7 @@ def self.apply(runner, model, spaces, hpxml_bldg, hpxml_header, schedules_file) fractions[[HPXML::LocationGarage, HPXML::LightingTypeLED]]) end end - grg_kwh = 0.0 if grg_kwh.nil? + grg_kwh = 0.0 if grg_kwh.nil? || n_occ == 0 grg_kwh *= lighting.garage_usage_multiplier unless lighting.garage_usage_multiplier.nil? # Add lighting to conditioned space diff --git a/HPXMLtoOpenStudio/resources/materials.rb b/HPXMLtoOpenStudio/resources/materials.rb index 19e7241b3a..484d79b4f2 100644 --- a/HPXMLtoOpenStudio/resources/materials.rb +++ b/HPXMLtoOpenStudio/resources/materials.rb @@ -272,31 +272,34 @@ def self.Concrete(thick_in) def self.ExteriorFinishMaterial(type, thick_in = nil) if (type == HPXML::SidingTypeNone) || (!thick_in.nil? && thick_in <= 0) return - elsif [HPXML::SidingTypeAsbestos].include? type + end + + case type + when HPXML::SidingTypeAsbestos thick_in = 0.25 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 4.20, rho: 118.6, cp: 0.24) - elsif [HPXML::SidingTypeBrick].include? type + when HPXML::SidingTypeBrick thick_in = 4.0 if thick_in.nil? return new(name: type, thick_in: thick_in, mat_base: BaseMaterial.Brick) - elsif [HPXML::SidingTypeCompositeShingle].include? type + when HPXML::SidingTypeCompositeShingle thick_in = 0.25 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 1.128, rho: 70.0, cp: 0.35) - elsif [HPXML::SidingTypeFiberCement].include? type + when HPXML::SidingTypeFiberCement thick_in = 0.375 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 1.79, rho: 21.7, cp: 0.24) - elsif [HPXML::SidingTypeMasonite].include? type # Masonite hardboard + when HPXML::SidingTypeMasonite # Masonite hardboard thick_in = 0.5 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 0.69, rho: 46.8, cp: 0.39) - elsif [HPXML::SidingTypeStucco].include? type + when HPXML::SidingTypeStucco thick_in = 1.0 if thick_in.nil? return new(name: type, thick_in: thick_in, mat_base: BaseMaterial.Stucco) - elsif [HPXML::SidingTypeSyntheticStucco].include? type # EIFS + when HPXML::SidingTypeSyntheticStucco # EIFS thick_in = 1.0 if thick_in.nil? return new(name: type, thick_in: thick_in, mat_base: BaseMaterial.InsulationRigid) - elsif [HPXML::SidingTypeVinyl, HPXML::SidingTypeAluminum].include? type + when HPXML::SidingTypeVinyl, HPXML::SidingTypeAluminum thick_in = 0.375 if thick_in.nil? return new(name: type, thick_in: thick_in, mat_base: BaseMaterial.Vinyl) - elsif [HPXML::SidingTypeWood].include? type + when HPXML::SidingTypeWood thick_in = 1.0 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 0.71, rho: 34.0, cp: 0.28) end @@ -310,25 +313,26 @@ def self.ExteriorFinishMaterial(type, thick_in = nil) # @param thick_in [TODO] TODO # @return [TODO] TODO def self.FoundationWallMaterial(type, thick_in) - if type == HPXML::FoundationWallTypeSolidConcrete + case type + when HPXML::FoundationWallTypeSolidConcrete return Material.Concrete(thick_in) - elsif type == HPXML::FoundationWallTypeDoubleBrick + when HPXML::FoundationWallTypeDoubleBrick return new(name: "#{type} #{thick_in} in.", thick_in: thick_in, mat_base: BaseMaterial.Brick, tAbs: 0.9) - elsif type == HPXML::FoundationWallTypeWood + when HPXML::FoundationWallTypeWood # Open wood cavity wall, so just assume 0.5" of sheathing return new(name: "#{type} #{thick_in} in.", thick_in: 0.5, mat_base: BaseMaterial.Wood, tAbs: 0.9) # Concrete block conductivity values below derived from Table 2 of # https://ncma.org/resource/rvalues-ufactors-of-single-wythe-concrete-masonry-walls/. Values # for 6-in thickness and 115 pcf, with interior/exterior films removed (R-0.68/R-0.17). - elsif type == HPXML::FoundationWallTypeConcreteBlockSolidCore + when HPXML::FoundationWallTypeConcreteBlockSolidCore return new(name: "#{type} #{thick_in} in.", thick_in: thick_in, k_in: 8.5, rho: 115.0, cp: 0.2, tAbs: 0.9) - elsif type == HPXML::FoundationWallTypeConcreteBlock + when HPXML::FoundationWallTypeConcreteBlock return new(name: "#{type} #{thick_in} in.", thick_in: thick_in, k_in: 5.0, rho: 45.0, cp: 0.2, tAbs: 0.9) - elsif type == HPXML::FoundationWallTypeConcreteBlockPerliteCore + when HPXML::FoundationWallTypeConcreteBlockPerliteCore return new(name: "#{type} #{thick_in} in.", thick_in: thick_in, k_in: 2.0, rho: 67.0, cp: 0.2, tAbs: 0.9) - elsif type == HPXML::FoundationWallTypeConcreteBlockFoamCore + when HPXML::FoundationWallTypeConcreteBlockFoamCore return new(name: "#{type} #{thick_in} in.", thick_in: thick_in, k_in: 1.8, rho: 67.0, cp: 0.2, tAbs: 0.9) - elsif type == HPXML::FoundationWallTypeConcreteBlockVermiculiteCore + when HPXML::FoundationWallTypeConcreteBlockVermiculiteCore return new(name: "#{type} #{thick_in} in.", thick_in: thick_in, k_in: 2.1, rho: 67.0, cp: 0.2, tAbs: 0.9) end @@ -345,11 +349,10 @@ def self.InteriorFinishMaterial(type, thick_in = nil) return else thick_in = 0.5 if thick_in.nil? - if [HPXML::InteriorFinishGypsumBoard, - HPXML::InteriorFinishGypsumCompositeBoard, - HPXML::InteriorFinishPlaster].include? type + case type + when HPXML::InteriorFinishGypsumBoard, HPXML::InteriorFinishGypsumCompositeBoard, HPXML::InteriorFinishPlaster return new(name: type, thick_in: thick_in, mat_base: BaseMaterial.Gypsum) - elsif [HPXML::InteriorFinishWood].include? type + when HPXML::InteriorFinishWood return new(name: type, thick_in: thick_in, mat_base: BaseMaterial.Wood) end end @@ -435,22 +438,23 @@ def self.RadiantBarrier(grade, is_attic_floor) # @param thick_in [TODO] TODO # @return [TODO] TODO def self.RoofMaterial(type, thick_in = nil) - if [HPXML::RoofTypeMetal].include? type + case type + when HPXML::RoofTypeMetal thick_in = 0.02 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 346.9, rho: 487.0, cp: 0.11) - elsif [HPXML::RoofTypeAsphaltShingles, HPXML::RoofTypeWoodShingles, HPXML::RoofTypeShingles, HPXML::RoofTypeCool].include? type + when HPXML::RoofTypeAsphaltShingles, HPXML::RoofTypeWoodShingles, HPXML::RoofTypeShingles, HPXML::RoofTypeCool thick_in = 0.25 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 1.128, rho: 70.0, cp: 0.35) - elsif [HPXML::RoofTypeConcrete].include? type + when HPXML::RoofTypeConcrete thick_in = 0.75 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 7.63, rho: 131.1, cp: 0.199) - elsif [HPXML::RoofTypeClayTile].include? type + when HPXML::RoofTypeClayTile thick_in = 0.75 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 5.83, rho: 118.6, cp: 0.191) - elsif [HPXML::RoofTypeEPS].include? type + when HPXML::RoofTypeEPS thick_in = 1.0 if thick_in.nil? return new(name: type, thick_in: thick_in, mat_base: BaseMaterial.InsulationRigid) - elsif [HPXML::RoofTypePlasticRubber].include? type + when HPXML::RoofTypePlasticRubber thick_in = 0.25 if thick_in.nil? return new(name: type, thick_in: thick_in, k_in: 2.78, rho: 110.8, cp: 0.36) end diff --git a/HPXMLtoOpenStudio/resources/misc_loads.rb b/HPXMLtoOpenStudio/resources/misc_loads.rb index 2e31b90403..53aa99f416 100644 --- a/HPXMLtoOpenStudio/resources/misc_loads.rb +++ b/HPXMLtoOpenStudio/resources/misc_loads.rb @@ -12,14 +12,20 @@ module MiscLoads # @param schedules_file [SchedulesFile] SchedulesFile wrapper class instance of detailed schedule files # @return [nil] def self.apply_plug_loads(runner, model, spaces, hpxml_bldg, hpxml_header, schedules_file) + if hpxml_bldg.building_occupancy.number_of_residents == 0 && (not hpxml_header.apply_ashrae140_assumptions) + # Operational calculation w/ zero occupants, zero out energy use + return + end + hpxml_bldg.plug_loads.each do |plug_load| - if plug_load.plug_load_type == HPXML::PlugLoadTypeOther + case plug_load.plug_load_type + when HPXML::PlugLoadTypeOther obj_name = Constants::ObjectTypeMiscPlugLoads - elsif plug_load.plug_load_type == HPXML::PlugLoadTypeTelevision + when HPXML::PlugLoadTypeTelevision obj_name = Constants::ObjectTypeMiscTelevision - elsif plug_load.plug_load_type == HPXML::PlugLoadTypeElectricVehicleCharging + when HPXML::PlugLoadTypeElectricVehicleCharging obj_name = Constants::ObjectTypeMiscElectricVehicleCharging - elsif plug_load.plug_load_type == HPXML::PlugLoadTypeWellPump + when HPXML::PlugLoadTypeWellPump obj_name = Constants::ObjectTypeMiscWellPump end if obj_name.nil? @@ -53,13 +59,14 @@ def self.apply_plug_load(runner, model, plug_load, obj_name, spaces, schedules_f # Create schedule sch = nil - if plug_load.plug_load_type == HPXML::PlugLoadTypeOther + case plug_load.plug_load_type + when HPXML::PlugLoadTypeOther col_name = SchedulesFile::Columns[:PlugLoadsOther].name - elsif plug_load.plug_load_type == HPXML::PlugLoadTypeTelevision + when HPXML::PlugLoadTypeTelevision col_name = SchedulesFile::Columns[:PlugLoadsTV].name - elsif plug_load.plug_load_type == HPXML::PlugLoadTypeElectricVehicleCharging + when HPXML::PlugLoadTypeElectricVehicleCharging col_name = SchedulesFile::Columns[:PlugLoadsVehicle].name - elsif plug_load.plug_load_type == HPXML::PlugLoadTypeWellPump + when HPXML::PlugLoadTypeWellPump col_name = SchedulesFile::Columns[:PlugLoadsWellPump].name end if not schedules_file.nil? @@ -111,11 +118,12 @@ def self.apply_plug_load(runner, model, plug_load, obj_name, spaces, schedules_f # @return [nil] def self.apply_fuel_loads(runner, model, spaces, hpxml_bldg, hpxml_header, schedules_file) hpxml_bldg.fuel_loads.each do |fuel_load| - if fuel_load.fuel_load_type == HPXML::FuelLoadTypeGrill + case fuel_load.fuel_load_type + when HPXML::FuelLoadTypeGrill obj_name = Constants::ObjectTypeMiscGrill - elsif fuel_load.fuel_load_type == HPXML::FuelLoadTypeLighting + when HPXML::FuelLoadTypeLighting obj_name = Constants::ObjectTypeMiscLighting - elsif fuel_load.fuel_load_type == HPXML::FuelLoadTypeFireplace + when HPXML::FuelLoadTypeFireplace obj_name = Constants::ObjectTypeMiscFireplace end if obj_name.nil? @@ -148,11 +156,12 @@ def self.apply_fuel_load(runner, model, fuel_load, obj_name, spaces, schedules_f # Create schedule sch = nil - if fuel_load.fuel_load_type == HPXML::FuelLoadTypeGrill + case fuel_load.fuel_load_type + when HPXML::FuelLoadTypeGrill col_name = SchedulesFile::Columns[:FuelLoadsGrill].name - elsif fuel_load.fuel_load_type == HPXML::FuelLoadTypeLighting + when HPXML::FuelLoadTypeLighting col_name = SchedulesFile::Columns[:FuelLoadsLighting].name - elsif fuel_load.fuel_load_type == HPXML::FuelLoadTypeFireplace + when HPXML::FuelLoadTypeFireplace col_name = SchedulesFile::Columns[:FuelLoadsFireplace].name end if not schedules_file.nil? diff --git a/HPXMLtoOpenStudio/resources/model.rb b/HPXMLtoOpenStudio/resources/model.rb index a30aacbbc5..5e2ba31032 100644 --- a/HPXMLtoOpenStudio/resources/model.rb +++ b/HPXMLtoOpenStudio/resources/model.rb @@ -353,6 +353,35 @@ def self.add_fan_system_model(model, name:, end_use:, power_per_flow:, max_flow_ return fan end + # Adds a PlantLoop object to the OpenStudio model. + # + # @param model [OpenStudio::Model::Model] OpenStudio Model object + # @param name [String] Name for the OpenStudio object + # @param fluid_type [String] Fluid type (Eplus::FluidXXX) + # @param glycol_concentration [Integer] Percent glycol concentration, only used if fluid is propylene glycol + # @param volume [Double] Volume of the entire loop, both demand and supply side (m^3) + # @param min_temp [Double] Minimum loop temperature (C) + # @param max_temp [Double] Maximum loop temperature (C) + # @param max_flow_rate [Double] Maximum loop flow rate (m^3/s) + # @return [OpenStudio::Model::PlantLoop] The model object + def self.add_plant_loop(model, name:, fluid_type: EPlus::FluidWater, glycol_concentration: 50, volume: nil, min_temp: nil, max_temp: nil, max_flow_rate: nil) + plant_loop = OpenStudio::Model::PlantLoop.new(model) + plant_loop.setName(name) + plant_loop.setFluidType(fluid_type) + if fluid_type == EPlus::FluidPropyleneGlycol + plant_loop.setGlycolConcentration(glycol_concentration) + end + plant_loop.setMinimumLoopTemperature(min_temp) unless min_temp.nil? + plant_loop.setMaximumLoopTemperature(max_temp) unless max_temp.nil? + plant_loop.setMaximumLoopFlowRate(max_flow_rate) unless max_flow_rate.nil? + if not volume.nil? + plant_loop.setPlantLoopVolume(volume) + else + plant_loop.autocalculatePlantLoopVolume() + end + return plant_loop + end + # Adds a PumpVariableSpeed object to the OpenStudio model. # # @param model [OpenStudio::Model::Model] OpenStudio Model object @@ -859,6 +888,8 @@ def self.reset(model, runner) # @return [nil] def self.merge_unit_models(model, hpxml_osm_map) # Map of OpenStudio IDD objects => OSM class names + # Note: ZoneCapacitanceMultiplierResearchSpecial is not actually unique, but we don't assign it to + # individual thermal zones, so we'll treat it as unique here. unique_object_map = { 'OS:ConvergenceLimits' => 'ConvergenceLimits', 'OS:Foundation:Kiva:Settings' => 'FoundationKivaSettings', 'OS:OutputControl:Files' => 'OutputControlFiles', @@ -875,7 +906,8 @@ def self.merge_unit_models(model, hpxml_osm_map) 'OS:Site:WaterMainsTemperature' => 'SiteWaterMainsTemperature', 'OS:SurfaceConvectionAlgorithm:Inside' => 'InsideSurfaceConvectionAlgorithm', 'OS:SurfaceConvectionAlgorithm:Outside' => 'OutsideSurfaceConvectionAlgorithm', - 'OS:Timestep' => 'Timestep' } + 'OS:Timestep' => 'Timestep', + 'OS:ZoneCapacitanceMultiplier:ResearchSpecial' => 'ZoneCapacitanceMultiplierResearchSpecial' } # Handle unique objects first: Grab one from the first model we find the # object on (may not be the first unit). diff --git a/HPXMLtoOpenStudio/resources/pv.rb b/HPXMLtoOpenStudio/resources/pv.rb index 5a9f60a2b1..08822275c7 100644 --- a/HPXMLtoOpenStudio/resources/pv.rb +++ b/HPXMLtoOpenStudio/resources/pv.rb @@ -64,22 +64,28 @@ def self.apply_pv_system(model, hpxml_bldg, pv_system) gpvwatts.setSystemLosses(pv_system.system_losses_fraction) gpvwatts.setTiltAngle(pv_system.array_tilt) gpvwatts.setAzimuthAngle(pv_system.array_azimuth) - if (pv_system.tracking == HPXML::PVTrackingTypeFixed) && (pv_system.location == HPXML::LocationRoof) - gpvwatts.setArrayType('FixedRoofMounted') - elsif (pv_system.tracking == HPXML::PVTrackingTypeFixed) && (pv_system.location == HPXML::LocationGround) - gpvwatts.setArrayType('FixedOpenRack') - elsif pv_system.tracking == HPXML::PVTrackingType1Axis + + case pv_system.tracking + when HPXML::PVTrackingTypeFixed + if pv_system.location == HPXML::LocationRoof + gpvwatts.setArrayType('FixedRoofMounted') + elsif pv_system.location == HPXML::LocationGround + gpvwatts.setArrayType('FixedOpenRack') + end + when HPXML::PVTrackingType1Axis gpvwatts.setArrayType('OneAxis') - elsif pv_system.tracking == HPXML::PVTrackingType1AxisBacktracked + when HPXML::PVTrackingType1AxisBacktracked gpvwatts.setArrayType('OneAxisBacktracking') - elsif pv_system.tracking == HPXML::PVTrackingType2Axis + when HPXML::PVTrackingType2Axis gpvwatts.setArrayType('TwoAxis') end - if pv_system.module_type == HPXML::PVModuleTypeStandard + + case pv_system.module_type + when HPXML::PVModuleTypeStandard gpvwatts.setModuleType('Standard') - elsif pv_system.module_type == HPXML::PVModuleTypePremium + when HPXML::PVModuleTypePremium gpvwatts.setModuleType('Premium') - elsif pv_system.module_type == HPXML::PVModuleTypeThinFilm + when HPXML::PVModuleTypeThinFilm gpvwatts.setModuleType('ThinFilm') end diff --git a/HPXMLtoOpenStudio/resources/schedules.rb b/HPXMLtoOpenStudio/resources/schedules.rb index f5ee36e47c..0da62c5f8a 100644 --- a/HPXMLtoOpenStudio/resources/schedules.rb +++ b/HPXMLtoOpenStudio/resources/schedules.rb @@ -1434,17 +1434,19 @@ def set_unavailable_periods(runner, unavailable_periods) @tmp_schedules.keys.each do |schedule_name| next if column_names.include? schedule_name - schedule_name2 = schedule_name - if [SchedulesFile::Columns[:HotWaterDishwasher].name].include?(schedule_name) + case schedule_name + when SchedulesFile::Columns[:HotWaterDishwasher].name schedule_name2 = SchedulesFile::Columns[:Dishwasher].name - elsif [SchedulesFile::Columns[:HotWaterClothesWasher].name].include?(schedule_name) + when SchedulesFile::Columns[:HotWaterClothesWasher].name schedule_name2 = SchedulesFile::Columns[:ClothesWasher].name - elsif [SchedulesFile::Columns[:HeatingSetpoint].name].include?(schedule_name) + when SchedulesFile::Columns[:HeatingSetpoint].name schedule_name2 = SchedulesFile::Columns[:SpaceHeating].name - elsif [SchedulesFile::Columns[:CoolingSetpoint].name].include?(schedule_name) + when SchedulesFile::Columns[:CoolingSetpoint].name schedule_name2 = SchedulesFile::Columns[:SpaceCooling].name - elsif [SchedulesFile::Columns[:WaterHeaterSetpoint].name].include?(schedule_name) + when SchedulesFile::Columns[:WaterHeaterSetpoint].name schedule_name2 = SchedulesFile::Columns[:WaterHeater].name + else + schedule_name2 = schedule_name end # Skip those unaffected diff --git a/HPXMLtoOpenStudio/resources/utility_bills.rb b/HPXMLtoOpenStudio/resources/utility_bills.rb index e9fba1f567..72b851bc8d 100644 --- a/HPXMLtoOpenStudio/resources/utility_bills.rb +++ b/HPXMLtoOpenStudio/resources/utility_bills.rb @@ -36,17 +36,19 @@ def self.get_rates_from_eia_data(runner, state_code, fuel_type, fixed_charge, ma end if not marginal_rate.nil? - if [HPXML::FuelTypeElectricity, HPXML::FuelTypeNaturalGas].include? fuel_type + case fuel_type + when HPXML::FuelTypeElectricity, HPXML::FuelTypeNaturalGas # Calculate average rate from user-specified marginal rate, user-specified fixed charge, and EIA data average_rate = marginal_rate_to_average_rate(marginal_rate, fixed_charge, household_consumption) - elsif [HPXML::FuelTypeOil, HPXML::FuelTypePropane, HPXML::FuelTypeCoal, HPXML::FuelTypeWoodCord, HPXML::FuelTypeWoodPellets].include? fuel_type + when HPXML::FuelTypeOil, HPXML::FuelTypePropane, HPXML::FuelTypeCoal, HPXML::FuelTypeWoodCord, HPXML::FuelTypeWoodPellets # Do nothing end else - if [HPXML::FuelTypeElectricity, HPXML::FuelTypeNaturalGas].include? fuel_type + case fuel_type + when HPXML::FuelTypeElectricity, HPXML::FuelTypeNaturalGas average_rate = get_eia_seds_rate(runner, state_code, fuel_type) marginal_rate = average_rate_to_marginal_rate(average_rate, fixed_charge, household_consumption) - elsif [HPXML::FuelTypeOil, HPXML::FuelTypePropane, HPXML::FuelTypeCoal, HPXML::FuelTypeWoodCord, HPXML::FuelTypeWoodPellets].include? fuel_type + when HPXML::FuelTypeOil, HPXML::FuelTypePropane, HPXML::FuelTypeCoal, HPXML::FuelTypeWoodCord, HPXML::FuelTypeWoodPellets marginal_rate = get_eia_seds_rate(runner, state_code, fuel_type) end end diff --git a/HPXMLtoOpenStudio/resources/version.rb b/HPXMLtoOpenStudio/resources/version.rb index 03fc5fb1a5..b56875f660 100644 --- a/HPXMLtoOpenStudio/resources/version.rb +++ b/HPXMLtoOpenStudio/resources/version.rb @@ -2,7 +2,7 @@ # Collection of methods related to software versions. module Version - OS_HPXML_Version = '1.9.0' # Version of the OS-HPXML workflow + OS_HPXML_Version = '1.10.0' # Version of the OS-HPXML workflow OS_Version = '3.9.0' # Required version of OpenStudio (can be 'X.X' or 'X.X.X') HPXML_Version = '4.0' # HPXML schemaVersion diff --git a/HPXMLtoOpenStudio/resources/waterheater.rb b/HPXMLtoOpenStudio/resources/waterheater.rb index 003b113a8f..3d9ec8a762 100644 --- a/HPXMLtoOpenStudio/resources/waterheater.rb +++ b/HPXMLtoOpenStudio/resources/waterheater.rb @@ -4,7 +4,7 @@ module Waterheater DefaultTankHeight = 4.0 # ft, assumption from BEopt - # TODO + # Adds any water heaters and appliances to the OpenStudio Model. # # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings # @param model [OpenStudio::Model::Model] OpenStudio Model object @@ -19,13 +19,14 @@ def self.apply_dhw_appliances(runner, model, weather, spaces, hpxml_bldg, hpxml_ plantloop_map = {} hpxml_bldg.water_heating_systems.each do |dhw_system| - if dhw_system.water_heater_type == HPXML::WaterHeaterTypeStorage + case dhw_system.water_heater_type + when HPXML::WaterHeaterTypeStorage apply_tank(model, runner, spaces, hpxml_bldg, hpxml_header, dhw_system, schedules_file, unavailable_periods, plantloop_map) - elsif dhw_system.water_heater_type == HPXML::WaterHeaterTypeTankless + when HPXML::WaterHeaterTypeTankless apply_tankless(model, runner, spaces, hpxml_bldg, hpxml_header, dhw_system, schedules_file, unavailable_periods, plantloop_map) - elsif dhw_system.water_heater_type == HPXML::WaterHeaterTypeHeatPump - apply_heatpump(model, runner, spaces, hpxml_bldg, hpxml_header, dhw_system, schedules_file, unavailable_periods, plantloop_map) - elsif [HPXML::WaterHeaterTypeCombiStorage, HPXML::WaterHeaterTypeCombiTankless].include? dhw_system.water_heater_type + when HPXML::WaterHeaterTypeHeatPump + apply_hpwh(model, runner, spaces, hpxml_bldg, hpxml_header, dhw_system, schedules_file, unavailable_periods, plantloop_map) + when HPXML::WaterHeaterTypeCombiStorage, HPXML::WaterHeaterTypeCombiTankless apply_combi(model, runner, spaces, hpxml_bldg, hpxml_header, dhw_system, schedules_file, unavailable_periods, plantloop_map) else fail "Unhandled water heater (#{dhw_system.water_heater_type})." @@ -35,12 +36,10 @@ def self.apply_dhw_appliances(runner, model, weather, spaces, hpxml_bldg, hpxml_ HotWaterAndAppliances.apply(runner, model, weather, spaces, hpxml_bldg, hpxml_header, schedules_file, plantloop_map) apply_solar_thermal(model, spaces, hpxml_bldg, plantloop_map) - - # Add combi-system EMS program with water use equipment information apply_combi_system_EMS(model, hpxml_bldg.water_heating_systems, plantloop_map) end - # TODO + # Adds a conventional storage tank water heater to the OpenStudio model. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings @@ -57,33 +56,33 @@ def self.apply_tank(model, runner, spaces, hpxml_bldg, hpxml_header, water_heati unit_multiplier = hpxml_bldg.building_construction.number_of_units solar_fraction = get_water_heater_solar_fraction(water_heating_system, hpxml_bldg) t_set_c = get_t_set_c(water_heating_system.temperature, water_heating_system.water_heater_type) - loop = create_new_loop(model, t_set_c, hpxml_header.eri_calculation_version, unit_multiplier) + plant_loop = add_plant_loop(model, t_set_c, hpxml_header.eri_calculation_version, unit_multiplier) act_vol = calc_storage_tank_actual_vol(water_heating_system.tank_volume, water_heating_system.fuel_type) - u, ua, eta_c = calc_tank_UA(act_vol, water_heating_system, solar_fraction, hpxml_bldg.building_construction.number_of_bedrooms) - new_heater = create_new_heater(name: Constants::ObjectTypeWaterHeater, - water_heating_system: water_heating_system, - act_vol: act_vol, - t_set_c: t_set_c, - loc_space: loc_space, - loc_schedule: loc_schedule, - model: model, - runner: runner, - u: u, - ua: ua, - eta_c: eta_c, - schedules_file: schedules_file, - unavailable_periods: unavailable_periods, - unit_multiplier: unit_multiplier) - loop.addSupplyBranchForComponent(new_heater) - - add_ec_adj(model, hpxml_bldg, new_heater, loc_space, water_heating_system, unit_multiplier) - add_desuperheater(model, runner, water_heating_system, new_heater, loc_space, loc_schedule, loop, unit_multiplier) - - plantloop_map[water_heating_system.id] = loop + u, ua, eta_c = disaggregate_tank_losses_and_burner_efficiency(act_vol, water_heating_system, solar_fraction, hpxml_bldg.building_construction.number_of_bedrooms) + water_heater = apply_water_heater(name: Constants::ObjectTypeWaterHeater, + water_heating_system: water_heating_system, + act_vol: act_vol, + t_set_c: t_set_c, + loc_space: loc_space, + loc_schedule: loc_schedule, + model: model, + runner: runner, + u: u, + ua: ua, + eta_c: eta_c, + schedules_file: schedules_file, + unavailable_periods: unavailable_periods, + unit_multiplier: unit_multiplier) + plant_loop.addSupplyBranchForComponent(water_heater) + + apply_ec_adj_program(model, hpxml_bldg, water_heater, loc_space, water_heating_system, unit_multiplier) + apply_desuperheater(model, runner, water_heating_system, water_heater, loc_space, loc_schedule, plant_loop, unit_multiplier) + + plantloop_map[water_heating_system.id] = plant_loop end - # TODO + # Adds a tankless water heater to the OpenStudio model. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings @@ -101,33 +100,33 @@ def self.apply_tankless(model, runner, spaces, hpxml_bldg, hpxml_header, water_h water_heating_system.heating_capacity = 100000000000.0 * unit_multiplier solar_fraction = get_water_heater_solar_fraction(water_heating_system, hpxml_bldg) t_set_c = get_t_set_c(water_heating_system.temperature, water_heating_system.water_heater_type) - loop = create_new_loop(model, t_set_c, hpxml_header.eri_calculation_version, unit_multiplier) + plant_loop = add_plant_loop(model, t_set_c, hpxml_header.eri_calculation_version, unit_multiplier) act_vol = 1.0 * unit_multiplier - _u, ua, eta_c = calc_tank_UA(act_vol, water_heating_system, solar_fraction, hpxml_bldg.building_construction.number_of_bedrooms) - new_heater = create_new_heater(name: Constants::ObjectTypeWaterHeater, - water_heating_system: water_heating_system, - act_vol: act_vol, - t_set_c: t_set_c, - loc_space: loc_space, - loc_schedule: loc_schedule, - model: model, - runner: runner, - ua: ua, - eta_c: eta_c, - schedules_file: schedules_file, - unavailable_periods: unavailable_periods, - unit_multiplier: unit_multiplier) - - loop.addSupplyBranchForComponent(new_heater) - - add_ec_adj(model, hpxml_bldg, new_heater, loc_space, water_heating_system, unit_multiplier) - add_desuperheater(model, runner, water_heating_system, new_heater, loc_space, loc_schedule, loop, unit_multiplier) - - plantloop_map[water_heating_system.id] = loop + _u, ua, eta_c = disaggregate_tank_losses_and_burner_efficiency(act_vol, water_heating_system, solar_fraction, hpxml_bldg.building_construction.number_of_bedrooms) + water_heater = apply_water_heater(name: Constants::ObjectTypeWaterHeater, + water_heating_system: water_heating_system, + act_vol: act_vol, + t_set_c: t_set_c, + loc_space: loc_space, + loc_schedule: loc_schedule, + model: model, + runner: runner, + ua: ua, + eta_c: eta_c, + schedules_file: schedules_file, + unavailable_periods: unavailable_periods, + unit_multiplier: unit_multiplier) + + plant_loop.addSupplyBranchForComponent(water_heater) + + apply_ec_adj_program(model, hpxml_bldg, water_heater, loc_space, water_heating_system, unit_multiplier) + apply_desuperheater(model, runner, water_heating_system, water_heater, loc_space, loc_schedule, plant_loop, unit_multiplier) + + plantloop_map[water_heating_system.id] = plant_loop end - # TODO + # Adds a heat pump water heater to the OpenStudio model. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings @@ -139,14 +138,13 @@ def self.apply_tankless(model, runner, spaces, hpxml_bldg, hpxml_header, water_h # @param unavailable_periods [HPXML::UnavailablePeriods] Object that defines periods for, e.g., power outages or vacancies # @param plantloop_map [Hash] Map of HPXML System ID => OpenStudio PlantLoop objects # @return [nil] - def self.apply_heatpump(model, runner, spaces, hpxml_bldg, hpxml_header, water_heating_system, schedules_file, unavailable_periods, plantloop_map) + def self.apply_hpwh(model, runner, spaces, hpxml_bldg, hpxml_header, water_heating_system, schedules_file, unavailable_periods, plantloop_map) loc_space, loc_schedule = Geometry.get_space_or_schedule_from_location(water_heating_system.location, model, spaces) unit_multiplier = hpxml_bldg.building_construction.number_of_units obj_name = Constants::ObjectTypeWaterHeater - conditioned_zone = spaces[HPXML::LocationConditionedSpace].thermalZone.get solar_fraction = get_water_heater_solar_fraction(water_heating_system, hpxml_bldg) t_set_c = get_t_set_c(water_heating_system.temperature, water_heating_system.water_heater_type) - loop = create_new_loop(model, t_set_c, hpxml_header.eri_calculation_version, unit_multiplier) + plant_loop = add_plant_loop(model, t_set_c, hpxml_header.eri_calculation_version, unit_multiplier) # Add in schedules for Tamb, RHamb, and the compressor hpwh_tamb = Model.add_schedule_constant( @@ -177,22 +175,22 @@ def self.apply_heatpump(model, runner, spaces, hpxml_bldg, hpxml_header, water_h limits: EPlus::ScheduleTypeLimitsTemperature ) - setpoint_schedule = nil + # To handle variable setpoints, need one schedule that gets sensed and a new schedule that gets actuated + sensed_setpoint_schedule = nil if not schedules_file.nil? - # To handle variable setpoints, need one schedule that gets sensed and a new schedule that gets actuated # Sensed schedule - setpoint_schedule = schedules_file.create_schedule_file(model, col_name: SchedulesFile::Columns[:WaterHeaterSetpoint].name, schedule_type_limits_name: EPlus::ScheduleTypeLimitsTemperature) - if not setpoint_schedule.nil? + sensed_setpoint_schedule = schedules_file.create_schedule_file(model, col_name: SchedulesFile::Columns[:WaterHeaterSetpoint].name, schedule_type_limits_name: EPlus::ScheduleTypeLimitsTemperature) + if not sensed_setpoint_schedule.nil? # Actuated schedule control_setpoint_schedule = ScheduleConstant.new(model, "#{obj_name} ControlSetpoint", 0.0, EPlus::ScheduleTypeLimitsTemperature, unavailable_periods: unavailable_periods) control_setpoint_schedule = control_setpoint_schedule.schedule end end - if setpoint_schedule.nil? - setpoint_schedule = ScheduleConstant.new(model, Constants::ObjectTypeWaterHeaterSetpoint, t_set_c, EPlus::ScheduleTypeLimitsTemperature, unavailable_periods: unavailable_periods) - setpoint_schedule = setpoint_schedule.schedule + if sensed_setpoint_schedule.nil? + sensed_setpoint_schedule = ScheduleConstant.new(model, Constants::ObjectTypeWaterHeaterSetpoint, t_set_c, EPlus::ScheduleTypeLimitsTemperature, unavailable_periods: unavailable_periods) + sensed_setpoint_schedule = sensed_setpoint_schedule.schedule - control_setpoint_schedule = setpoint_schedule + control_setpoint_schedule = sensed_setpoint_schedule else runner.registerWarning("Both '#{SchedulesFile::Columns[:WaterHeaterSetpoint].name}' schedule file and setpoint temperature provided; the latter will be ignored.") if !t_set_c.nil? end @@ -202,13 +200,13 @@ def self.apply_heatpump(model, runner, spaces, hpxml_bldg, hpxml_header, water_h max_temp = 120.0 # F # Coil:WaterHeating:AirToWaterHeatPump:Wrapped - coil = setup_hpwh_dxcoil(model, runner, water_heating_system, hpxml_bldg.elevation, obj_name, airflow_rate, unit_multiplier) + coil = apply_hpwh_dxcoil(model, runner, water_heating_system, hpxml_bldg.elevation, obj_name, airflow_rate, unit_multiplier) # WaterHeater:Stratified - tank = setup_hpwh_stratified_tank(model, water_heating_system, obj_name, solar_fraction, hpwh_tamb, bottom_element_sp, top_element_sp, unit_multiplier, hpxml_bldg.building_construction.number_of_bedrooms) - loop.addSupplyBranchForComponent(tank) + tank = apply_hpwh_stratified_tank(model, water_heating_system, obj_name, solar_fraction, hpwh_tamb, top_element_sp, bottom_element_sp, unit_multiplier, hpxml_bldg.building_construction.number_of_bedrooms) + plant_loop.addSupplyBranchForComponent(tank) - add_desuperheater(model, runner, water_heating_system, tank, loc_space, loc_schedule, loop, unit_multiplier) + apply_desuperheater(model, runner, water_heating_system, tank, loc_space, loc_schedule, plant_loop, unit_multiplier) # Fan:SystemModel fan_power = 0.0462 # W/cfm, Based on 1st gen AO Smith HPWH, could be updated but pretty minor impact @@ -223,30 +221,33 @@ def self.apply_heatpump(model, runner, spaces, hpxml_bldg, hpxml_header, water_h fan.additionalProperties.setFeature('ObjectType', Constants::ObjectTypeWaterHeater) # Used by reporting measure # WaterHeater:HeatPump:WrappedCondenser - hpwh = setup_hpwh_wrapped_condenser(model, obj_name, coil, tank, fan, airflow_rate, hpwh_tamb, hpwh_rhamb, min_temp, max_temp, control_setpoint_schedule, unit_multiplier) + hpwh = apply_hpwh_wrapped_condenser(model, obj_name, coil, tank, fan, airflow_rate, hpwh_tamb, hpwh_rhamb, min_temp, max_temp, control_setpoint_schedule, unit_multiplier) # Amb temp & RH sensors, temp sensor shared across programs - amb_temp_sensor, amb_rh_sensors = get_loc_temp_rh_sensors(model, obj_name, loc_space, loc_schedule, conditioned_zone) - hpwh_inlet_air_program = add_hpwh_inlet_air_and_zone_heat_gain_program(model, obj_name, loc_space, hpwh_tamb, hpwh_rhamb, tank, coil, fan, amb_temp_sensor, amb_rh_sensors, unit_multiplier) + amb_temp_sensor, amb_rh_sensors = apply_hpwh_loc_temp_rh_sensors(model, obj_name, loc_space, loc_schedule, spaces) + hpwh_zone_heat_gain_program = apply_hpwh_zone_heat_gain_program(model, obj_name, loc_space, hpwh_tamb, hpwh_rhamb, tank, coil, fan, amb_temp_sensor, amb_rh_sensors, unit_multiplier) # EMS for the HPWH control logic - op_mode = water_heating_system.operating_mode - hpwh_ctrl_program = add_hpwh_control_program(model, runner, obj_name, amb_temp_sensor, top_element_sp, bottom_element_sp, min_temp, max_temp, op_mode, setpoint_schedule, control_setpoint_schedule, schedules_file) + hpwh_ctrl_program = apply_hpwh_control_program(model, runner, obj_name, water_heating_system, amb_temp_sensor, top_element_sp, bottom_element_sp, min_temp, max_temp, sensed_setpoint_schedule, control_setpoint_schedule, schedules_file) # ProgramCallingManagers Model.add_ems_program_calling_manager( model, name: "#{obj_name} ProgramManager", calling_point: 'InsideHVACSystemIterationLoop', - ems_programs: [hpwh_ctrl_program, hpwh_inlet_air_program] + ems_programs: [hpwh_ctrl_program, hpwh_zone_heat_gain_program] ) - add_ec_adj(model, hpxml_bldg, hpwh, loc_space, water_heating_system, unit_multiplier) + apply_ec_adj_program(model, hpxml_bldg, hpwh, loc_space, water_heating_system, unit_multiplier) - plantloop_map[water_heating_system.id] = loop + plantloop_map[water_heating_system.id] = plant_loop end - # TODO + # Adds a combination boiler water heater to the OpenStudio model. + # + # Note that we model two separate boilers (one for just space heating and one for just water heating). + # This makes the code much simpler and makes the EnergyPlus results more robust. + # # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings @@ -277,35 +278,35 @@ def self.apply_combi(model, runner, spaces, hpxml_bldg, hpxml_header, water_heat act_vol = calc_storage_tank_actual_vol(water_heating_system.tank_volume, nil) a_side = calc_tank_areas(act_vol)[1] - ua = calc_indirect_ua_with_standbyloss(act_vol, water_heating_system, a_side, solar_fraction, hpxml_bldg.building_construction.number_of_bedrooms) + ua = calc_combi_tank_losses(act_vol, water_heating_system, a_side, solar_fraction, hpxml_bldg.building_construction.number_of_bedrooms) else ua = 0.0 act_vol = 1.0 end t_set_c = get_t_set_c(water_heating_system.temperature, water_heating_system.water_heater_type) - loop = create_new_loop(model, t_set_c, hpxml_header.eri_calculation_version, unit_multiplier) + plant_loop = add_plant_loop(model, t_set_c, hpxml_header.eri_calculation_version, unit_multiplier) # Create water heater - new_heater = create_new_heater(name: obj_name_combi, - water_heating_system: water_heating_system, - act_vol: act_vol, - t_set_c: t_set_c, - loc_space: loc_space, - loc_schedule: loc_schedule, - model: model, - runner: runner, - ua: ua, - is_combi: true, - schedules_file: schedules_file, - unavailable_periods: unavailable_periods, - unit_multiplier: unit_multiplier) - new_heater.setSourceSideDesignFlowRate(100 * unit_multiplier) # set one large number, override by EMS + water_heater = apply_water_heater(name: obj_name_combi, + water_heating_system: water_heating_system, + act_vol: act_vol, + t_set_c: t_set_c, + loc_space: loc_space, + loc_schedule: loc_schedule, + model: model, + runner: runner, + ua: ua, + is_combi: true, + schedules_file: schedules_file, + unavailable_periods: unavailable_periods, + unit_multiplier: unit_multiplier) + water_heater.setSourceSideDesignFlowRate(100 * unit_multiplier) # set one large number, override by EMS # Create alternate setpoint schedule for source side flow request - alternate_stp_sch = new_heater.setpointTemperatureSchedule.get.clone(model).to_Schedule.get + alternate_stp_sch = water_heater.setpointTemperatureSchedule.get.clone(model).to_Schedule.get alternate_stp_sch.setName("#{obj_name_combi} Alt Spt") - new_heater.setIndirectAlternateSetpointTemperatureSchedule(alternate_stp_sch) + water_heater.setIndirectAlternateSetpointTemperatureSchedule(alternate_stp_sch) # Create setpoint schedule to specify source side temperature # tank source side inlet temperature, degree C @@ -323,28 +324,31 @@ def self.apply_combi(model, runner, spaces, hpxml_bldg, hpxml_header, water_heat # change loop equipment operation scheme to heating load scheme_dhw = OpenStudio::Model::PlantEquipmentOperationHeatingLoad.new(model) - scheme_dhw.addEquipment(1000000000, new_heater) - loop.setPrimaryPlantEquipmentOperationScheme(scheme_dhw) + scheme_dhw.addEquipment(1000000000, water_heater) + plant_loop.setPrimaryPlantEquipmentOperationScheme(scheme_dhw) # Add dhw boiler to the load distribution scheme scheme = OpenStudio::Model::PlantEquipmentOperationHeatingLoad.new(model) scheme.addEquipment(1000000000, boiler) boiler_plant_loop.setPrimaryPlantEquipmentOperationScheme(scheme) - boiler_plant_loop.addDemandBranchForComponent(new_heater) + boiler_plant_loop.addDemandBranchForComponent(water_heater) boiler_plant_loop.setPlantLoopVolume(0.001 * unit_multiplier) # Cannot be auto-calculated because of large default tank source side mfr(set to be overwritten by EMS) - loop.addSupplyBranchForComponent(new_heater) + plant_loop.addSupplyBranchForComponent(water_heater) - add_ec_adj(model, hpxml_bldg, new_heater, loc_space, water_heating_system, unit_multiplier, boiler) + apply_ec_adj_program(model, hpxml_bldg, water_heater, loc_space, water_heating_system, unit_multiplier, boiler) - plantloop_map[water_heating_system.id] = loop + plantloop_map[water_heating_system.id] = plant_loop end - # TODO + # Calculates the water heating energy consumption adjustment factor (EC_adj) due to the effectiveness of + # the hot water distribution system. + # + # Source: ANSI/RESNET/ICC 301 # # @param hpxml_bldg [HPXML::Building] HPXML Building object representing an individual dwelling unit # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest - # @return [TODO] TODO + # @return [Double] Hot water energy consumption multiplier def self.get_dist_energy_consumption_adjustment(hpxml_bldg, water_heating_system) if water_heating_system.fraction_dhw_load_served <= 0 # No fixtures; not accounting for distribution system @@ -358,9 +362,7 @@ def self.get_dist_energy_consumption_adjustment(hpxml_bldg, water_heating_system cfa = hpxml_bldg.building_construction.conditioned_floor_area ncfl = hpxml_bldg.building_construction.number_of_conditioned_floors - # ANSI/RESNET 301-2014 Addendum A-2015 - # Amendment on Domestic Hot Water (DHW) Systems - # Eq. 4.2-16 + # ANSI/RESNET/ICC 301-2022 Eq. 4.2-44 ew_fact = get_dist_energy_waste_factor(hot_water_distribution) o_frac = 0.25 # fraction of hot water waste from standard operating conditions oew_fact = ew_fact * o_frac # standard operating condition portion of hot water energy waste @@ -377,35 +379,35 @@ def self.get_dist_energy_consumption_adjustment(hpxml_bldg, water_heating_system return (e_waste + 128.0) / 160.0 end - # TODO + # Retrieves the hot water distribution system relative annual energy waste factors. + # + # Source: ANSI/RESNET/ICC 301 # - # @param hot_water_distribution [TODO] TODO - # @return [TODO] TODO + # @param hot_water_distribution [HPXML::HotWaterDistribution] The HPXML hot water distribution system of interest + # @return [Double] Energy waste factor def self.get_dist_energy_waste_factor(hot_water_distribution) - # ANSI/RESNET 301-2014 Addendum A-2015 - # Amendment on Domestic Hot Water (DHW) Systems - # Table 4.2.2.5.2.11(6) Hot water distribution system relative annual energy waste factors + # ANSI/RESNET/ICC 301-2022 Table 4.2.2.7.2.11(6) if hot_water_distribution.system_type == HPXML::DHWDistTypeRecirc - if (hot_water_distribution.recirculation_control_type == HPXML::DHWRecircControlTypeNone) || - (hot_water_distribution.recirculation_control_type == HPXML::DHWRecircControlTypeTimer) + case hot_water_distribution.recirculation_control_type + when HPXML::DHWRecircControlTypeNone, HPXML::DHWRecircControlTypeTimer if hot_water_distribution.pipe_r_value < 3.0 return 500.0 else return 250.0 end - elsif hot_water_distribution.recirculation_control_type == HPXML::DHWRecircControlTypeTemperature + when HPXML::DHWRecircControlTypeTemperature if hot_water_distribution.pipe_r_value < 3.0 return 375.0 else return 187.5 end - elsif hot_water_distribution.recirculation_control_type == HPXML::DHWRecircControlTypeSensor + when HPXML::DHWRecircControlTypeSensor if hot_water_distribution.pipe_r_value < 3.0 return 64.8 else return 43.2 end - elsif hot_water_distribution.recirculation_control_type == HPXML::DHWRecircControlTypeManual + when HPXML::DHWRecircControlTypeManual if hot_water_distribution.pipe_r_value < 3.0 return 43.2 else @@ -422,7 +424,9 @@ def self.get_dist_energy_waste_factor(hot_water_distribution) fail 'Unexpected hot water distribution system.' end - # TODO + # Adds an EMS program to control the boiler operation based on domestic hot water demand. + # The program modulates the source side mass flow rate to achieve better control and accuracy + # compared to not having the EMS program. See https://github.com/NREL/OpenStudio-HPXML/pull/225. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param water_heating_systems [Array] The HPXML water heaters of interest @@ -594,7 +598,7 @@ def self.apply_combi_system_EMS(model, water_heating_systems, plantloop_map) end end - # TODO + # Adds an HPXML Solar Thermal System to the OpenStudio model. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param spaces [Hash] Map of HPXML locations => OpenStudio Space objects @@ -620,21 +624,23 @@ def self.apply_solar_thermal(model, spaces, hpxml_bldg, plantloop_map) obj_name = Constants::ObjectTypeSolarHotWater - if [HPXML::SolarThermalCollectorTypeEvacuatedTube].include? solar_thermal_system.collector_type + case solar_thermal_system.collector_type + when HPXML::SolarThermalCollectorTypeEvacuatedTube iam_coeff2 = 0.3023 # IAM coeff1=1 by definition, values based on a system listed by SRCC with values close to the average iam_coeff3 = -0.3057 - elsif [HPXML::SolarThermalCollectorTypeSingleGlazing, HPXML::SolarThermalCollectorTypeDoubleGlazing].include? solar_thermal_system.collector_type + when HPXML::SolarThermalCollectorTypeSingleGlazing, HPXML::SolarThermalCollectorTypeDoubleGlazing iam_coeff2 = 0.1 iam_coeff3 = 0 - elsif [HPXML::SolarThermalCollectorTypeICS].include? solar_thermal_system.collector_type + when HPXML::SolarThermalCollectorTypeICS iam_coeff2 = 0.1 iam_coeff3 = 0 end - if [HPXML::SolarThermalLoopTypeIndirect].include? solar_thermal_system.collector_loop_type + case solar_thermal_system.collector_loop_type + when HPXML::SolarThermalLoopTypeIndirect fluid_type = EPlus::FluidPropyleneGlycol heat_ex_eff = 0.7 - elsif [HPXML::SolarThermalLoopTypeDirect, HPXML::SolarThermalLoopTypeThermosyphon].include? solar_thermal_system.collector_loop_type + when HPXML::SolarThermalLoopTypeDirect, HPXML::SolarThermalLoopTypeThermosyphon fluid_type = EPlus::FluidWater heat_ex_eff = 1.0 end @@ -680,19 +686,11 @@ def self.apply_solar_thermal(model, spaces, hpxml_bldg, plantloop_map) end end - plant_loop = OpenStudio::Model::PlantLoop.new(model) - plant_loop.setName('solar hot water loop') - if fluid_type == EPlus::FluidWater - plant_loop.setFluidType(EPlus::FluidWater) - else - plant_loop.setFluidType(EPlus::FluidPropyleneGlycol) - plant_loop.setGlycolConcentration(50) - end - plant_loop.setMaximumLoopTemperature(100) - plant_loop.setMinimumLoopTemperature(0) - plant_loop.setMinimumLoopFlowRate(0) - plant_loop.setLoadDistributionScheme('Optimal') - plant_loop.setPlantEquipmentOperationHeatingLoadSchedule(model.alwaysOnDiscreteSchedule) + plant_loop = Model.add_plant_loop( + model, + name: 'solar hot water loop', + fluid_type: fluid_type + ) sizing_plant = plant_loop.sizingPlant sizing_plant.setLoopType('Heating') @@ -824,7 +822,7 @@ def self.apply_solar_thermal(model, spaces, hpxml_bldg, plantloop_map) storage_tank.setHeaterFuelType(EPlus::FuelTypeElectricity) storage_tank.setHeaterThermalEfficiency(1) storage_tank.ambientTemperatureSchedule.get.remove - set_wh_ambient(loc_space, loc_schedule, storage_tank) + apply_ambient_temperature(loc_space, loc_schedule, storage_tank) storage_tank.setSkinLossFractiontoZone(1.0 / unit_multiplier) # Tank losses are multiplied by E+ zone multiplier, so need to compensate here storage_tank.setOffCycleFlueLossFractiontoZone(1.0 / unit_multiplier) storage_tank.setUseSideEffectiveness(1) @@ -838,7 +836,7 @@ def self.apply_solar_thermal(model, spaces, hpxml_bldg, plantloop_map) storage_tank.setOnCycleParasiticFuelConsumptionRate(0) storage_tank.setOffCycleParasiticFuelConsumptionRate(0) storage_tank.setUseSideDesignFlowRate(UnitConversions.convert(storage_volume, 'gal', 'm^3') / 60.1) # Sized to ensure that E+ never autosizes the design flow rate to be larger than the tank volume getting drawn out in a hour (60 minutes) - set_stratified_tank_ua(storage_tank, u_tank, unit_multiplier) + apply_stratified_tank_losses(storage_tank, u_tank, unit_multiplier) storage_tank.additionalProperties.setFeature('HPXML_ID', solar_thermal_system.water_heating_system.id) # Used by reporting measure storage_tank.additionalProperties.setFeature('ObjectType', Constants::ObjectTypeSolarHotWater) # Used by reporting measure @@ -897,23 +895,23 @@ def self.apply_solar_thermal(model, spaces, hpxml_bldg, plantloop_map) ) end - # TODO + # Adds a WaterHeaterHeatPumpWrappedCondenser object for the HPWH to the OpenStudio model. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param obj_name [String] Name for the OpenStudio object - # @param coil [TODO] TODO - # @param tank [TODO] TODO - # @param fan [TODO] TODO - # @param airflow_rate [TODO] TODO - # @param hpwh_tamb [TODO] TODO - # @param hpwh_rhamb [TODO] TODO - # @param min_temp [TODO] TODO - # @param max_temp [TODO] TODO - # @param setpoint_schedule [TODO] TODO + # @param coil [OpenStudio::Model::CoilWaterHeatingAirToWaterHeatPumpWrapped] The HPWH DX coil + # @param tank [OpenStudio::Model::WaterHeaterStratified] The HPWH storage tank + # @param fan [OpenStudio::Model::FanSystemModel] The HPWH fan + # @param airflow_rate [Double] HPWH fan airflow rate (cfm) + # @param hpwh_tamb [OpenStudio::Model::ScheduleConstant] HPWH ambient temperature (C) + # @param hpwh_rhamb [OpenStudio::Model::ScheduleConstant] HPWH ambient relative humidity + # @param min_temp [Double] Minimum temperature for compressor operation (F) + # @param max_temp [Double] Maximum temperature for compressor operation (F) + # @param control_setpoint_schedule [OpenStudio::Model::ScheduleConstant or OpenStudio::Model::ScheduleRuleset] Setpoint temperature schedule (controlled) # @param unit_multiplier [Integer] Number of similar dwelling units - # @return [TODO] TODO - def self.setup_hpwh_wrapped_condenser(model, obj_name, coil, tank, fan, airflow_rate, hpwh_tamb, hpwh_rhamb, min_temp, max_temp, setpoint_schedule, unit_multiplier) - hpwh = OpenStudio::Model::WaterHeaterHeatPumpWrappedCondenser.new(model, coil, tank, fan, setpoint_schedule, model.alwaysOnDiscreteSchedule) + # @return [OpenStudio::Model::WaterHeaterHeatPumpWrappedCondenser] The HPWH object + def self.apply_hpwh_wrapped_condenser(model, obj_name, coil, tank, fan, airflow_rate, hpwh_tamb, hpwh_rhamb, min_temp, max_temp, control_setpoint_schedule, unit_multiplier) + hpwh = OpenStudio::Model::WaterHeaterHeatPumpWrappedCondenser.new(model, coil, tank, fan, control_setpoint_schedule, model.alwaysOnDiscreteSchedule) hpwh.setName("#{obj_name} hpwh") hpwh.setDeadBandTemperatureDifference(3.89) hpwh.setCondenserBottomLocation((1.0 - (12 - 0.5) / 12.0) * tank.tankHeight.get) # in the 12th node of a 12-node tank (counting from top) @@ -938,17 +936,17 @@ def self.setup_hpwh_wrapped_condenser(model, obj_name, coil, tank, fan, airflow_ return hpwh end - # TODO + # Adds a CoilWaterHeatingAirToWaterHeatPumpWrapped object for the HPWH to the OpenStudio model. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest # @param elevation [Double] Elevation of the building site (ft) # @param obj_name [String] Name for the OpenStudio object - # @param airflow_rate [TODO] TODO + # @param airflow_rate [Double] HPWH fan airflow rate (cfm) # @param unit_multiplier [Integer] Number of similar dwelling units - # @return [TODO] TODO - def self.setup_hpwh_dxcoil(model, runner, water_heating_system, elevation, obj_name, airflow_rate, unit_multiplier) + # @return [OpenStudio::Model::CoilWaterHeatingAirToWaterHeatPumpWrapped] The HPWH DX coil + def self.apply_hpwh_dxcoil(model, runner, water_heating_system, elevation, obj_name, airflow_rate, unit_multiplier) # Curves hpwh_cap = Model.add_curve_biquadratic( model, @@ -1011,32 +1009,33 @@ def self.set_heat_pump_cop(water_heating_system) cop = 1.174536058 * uef # Based on simulation of the UEF test procedure at varying COPs elsif not water_heating_system.uniform_energy_factor.nil? uef = water_heating_system.uniform_energy_factor - if water_heating_system.usage_bin == HPXML::WaterHeaterUsageBinVerySmall + case water_heating_system.usage_bin + when HPXML::WaterHeaterUsageBinVerySmall fail 'It is unlikely that a heat pump water heater falls into the very small bin of the First Hour Rating (FHR) test. Double check input.' - elsif water_heating_system.usage_bin == HPXML::WaterHeaterUsageBinLow + when HPXML::WaterHeaterUsageBinLow cop = 1.0005 * uef - 0.0789 - elsif water_heating_system.usage_bin == HPXML::WaterHeaterUsageBinMedium + when HPXML::WaterHeaterUsageBinMedium cop = 1.0909 * uef - 0.0868 - elsif water_heating_system.usage_bin == HPXML::WaterHeaterUsageBinHigh + when HPXML::WaterHeaterUsageBinHigh cop = 1.1022 * uef - 0.0877 end end water_heating_system.additional_properties.cop = cop end - # TODO + # Adds a WaterHeaterStratified object for the HPWH to the OpenStudio model. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest # @param obj_name [String] Name for the OpenStudio object - # @param solar_fraction [TODO] TODO - # @param hpwh_tamb [TODO] TODO - # @param hpwh_bottom_element_sp [TODO] TODO - # @param hpwh_top_element_sp [TODO] TODO + # @param solar_fraction [Double] Portion of hot water load served by an attached solar thermal system + # @param hpwh_tamb [OpenStudio::Model::ScheduleConstant] HPWH ambient temperature (C) + # @param hpwh_top_element_sp [OpenStudio::Model::ScheduleConstant] HPWH top element setpoint schedule + # @param hpwh_bottom_element_sp [OpenStudio::Model::ScheduleConstant] HPWH bottom element setpoint schedule # @param unit_multiplier [Integer] Number of similar dwelling units # @param nbeds [Integer] Number of bedrooms in the dwelling unit - # @return [TODO] TODO - def self.setup_hpwh_stratified_tank(model, water_heating_system, obj_name, solar_fraction, hpwh_tamb, hpwh_bottom_element_sp, hpwh_top_element_sp, unit_multiplier, nbeds) + # @return [OpenStudio::Model::WaterHeaterStratified] The HPWH storage tank + def self.apply_hpwh_stratified_tank(model, water_heating_system, obj_name, solar_fraction, hpwh_tamb, hpwh_top_element_sp, hpwh_bottom_element_sp, unit_multiplier, nbeds) h_tank = 0.0188 * water_heating_system.tank_volume + 0.0935 # m; Linear relationship that gets GE height at 50 gal and AO Smith height at 80 gal # Calculate some geometry parameters for UA, the location of sensors and heat sources in the tank @@ -1095,21 +1094,23 @@ def self.setup_hpwh_stratified_tank(model, water_heating_system, obj_name, solar tank.setSourceSideOutletHeight(0) tank.setSkinLossFractiontoZone(1.0 / unit_multiplier) # Tank losses are multiplied by E+ zone multiplier, so need to compensate here tank.setOffCycleFlueLossFractiontoZone(1.0 / unit_multiplier) - set_stratified_tank_ua(tank, u_tank, unit_multiplier) + apply_stratified_tank_losses(tank, u_tank, unit_multiplier) tank.additionalProperties.setFeature('HPXML_ID', water_heating_system.id) # Used by reporting measure return tank end - # TODO + # Adds EMS sensors for the HPWH ambient temperature and ambient relative humidity. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param obj_name [String] Name for the OpenStudio object # @param loc_space [OpenStudio::Model::Space] The space where the water heater is located - # @param loc_schedule [OpenStudio::Model::ScheduleConstant] The temperature schedule for where the water heater is located, if not in a space - # @param conditioned_zone [TODO] TODO - # @return [TODO] TODO - def self.get_loc_temp_rh_sensors(model, obj_name, loc_space, loc_schedule, conditioned_zone) + # @param loc_schedule [OpenStudio::Model::ScheduleConstant] The temperature schedule, if not located in a space + # @param spaces [Hash] Map of HPXML locations => OpenStudio Space objects + # @return [Array>] HPWH ambient temperature sensor, One or more HPWH ambient RH sensors + def self.apply_hpwh_loc_temp_rh_sensors(model, obj_name, loc_space, loc_schedule, spaces) + conditioned_zone = spaces[HPXML::LocationConditionedSpace].thermalZone.get + rh_sensors = [] if not loc_schedule.nil? amb_temp_sensor = Model.add_ems_sensor( @@ -1179,21 +1180,21 @@ def self.get_loc_temp_rh_sensors(model, obj_name, loc_space, loc_schedule, condi return amb_temp_sensor, rh_sensors end - # TODO + # Adds an EMS program to add sensible/latent gains to the thermal zone where the HPWH is located. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param obj_name [String] Name for the OpenStudio object # @param loc_space [OpenStudio::Model::Space] The space where the water heater is located - # @param hpwh_tamb [TODO] TODO - # @param hpwh_rhamb [TODO] TODO - # @param tank [TODO] TODO - # @param coil [TODO] TODO - # @param fan [TODO] TODO - # @param amb_temp_sensor [TODO] TODO - # @param amb_rh_sensors [TODO] TODO + # @param hpwh_tamb [OpenStudio::Model::ScheduleConstant] HPWH ambient temperature (C) + # @param hpwh_rhamb [OpenStudio::Model::ScheduleConstant] HPWH ambient relative humidity + # @param tank [OpenStudio::Model::WaterHeaterStratified] The HPWH storage tank + # @param coil [OpenStudio::Model::CoilWaterHeatingAirToWaterHeatPumpWrapped] The HPWH DX coil + # @param fan [OpenStudio::Model::FanSystemModel] The HPWH fan + # @param amb_temp_sensor [OpenStudio::Model::EnergyManagementSystemSensor] HPWH ambient temperature sensor + # @param amb_rh_sensors [Array] One or more HPWH ambient RH sensors # @param unit_multiplier [Integer] Number of similar dwelling units - # @return [TODO] TODO - def self.add_hpwh_inlet_air_and_zone_heat_gain_program(model, obj_name, loc_space, hpwh_tamb, hpwh_rhamb, tank, coil, fan, amb_temp_sensor, amb_rh_sensors, unit_multiplier) + # @return [OpenStudio::Model::EnergyManagementSystemProgram] The HPWH heat gain program + def self.apply_hpwh_zone_heat_gain_program(model, obj_name, loc_space, hpwh_tamb, hpwh_rhamb, tank, coil, fan, amb_temp_sensor, amb_rh_sensors, unit_multiplier) # EMS Actuators: Inlet T & RH, sensible and latent gains to the space tamb_act_actuator = Model.add_ems_actuator( name: "#{obj_name} Tamb act", @@ -1275,41 +1276,41 @@ def self.add_hpwh_inlet_air_and_zone_heat_gain_program(model, obj_name, loc_spac key_name: fan.name ) - hpwh_inlet_air_program = Model.add_ems_program( + hpwh_zone_heat_gain_program = Model.add_ems_program( model, name: "#{obj_name} InletAir" ) - hpwh_inlet_air_program.addLine("Set #{tamb_act_actuator.name} = #{amb_temp_sensor.name}") + hpwh_zone_heat_gain_program.addLine("Set #{tamb_act_actuator.name} = #{amb_temp_sensor.name}") # Average relative humidity for mf spaces: other multifamily buffer space & other heated space - hpwh_inlet_air_program.addLine("Set #{rhamb_act_actuator.name} = 0") + hpwh_zone_heat_gain_program.addLine("Set #{rhamb_act_actuator.name} = 0") amb_rh_sensors.each do |amb_rh_sensor| - hpwh_inlet_air_program.addLine("Set #{rhamb_act_actuator.name} = #{rhamb_act_actuator.name} + (#{amb_rh_sensor.name} / 100) / #{amb_rh_sensors.size}") + hpwh_zone_heat_gain_program.addLine("Set #{rhamb_act_actuator.name} = #{rhamb_act_actuator.name} + (#{amb_rh_sensor.name} / 100) / #{amb_rh_sensors.size}") end if not loc_space.nil? # Sensible/latent heat gain to the space # Tank losses are multiplied by E+ zone multiplier, so need to compensate here - hpwh_inlet_air_program.addLine("Set #{sens_act_actuator.name} = (0 - #{sens_cool_sensor.name} - (#{tl_sensor.name} + #{fan_power_sensor.name})) / #{unit_multiplier}") - hpwh_inlet_air_program.addLine("Set #{lat_act_actuator.name} = (0 - #{lat_cool_sensor.name}) / #{unit_multiplier}") + hpwh_zone_heat_gain_program.addLine("Set #{sens_act_actuator.name} = (0 - #{sens_cool_sensor.name} - (#{tl_sensor.name} + #{fan_power_sensor.name})) / #{unit_multiplier}") + hpwh_zone_heat_gain_program.addLine("Set #{lat_act_actuator.name} = (0 - #{lat_cool_sensor.name}) / #{unit_multiplier}") end - return hpwh_inlet_air_program + return hpwh_zone_heat_gain_program end - # TODO + # Adds an EMS program to control the HPWH upper and lower elements. # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings # @param obj_name [String] Name for the OpenStudio object - # @param amb_temp_sensor [TODO] TODO - # @param hpwh_top_element_sp [TODO] TODO - # @param hpwh_bottom_element_sp [TODO] TODO - # @param min_temp [TODO] TODO - # @param max_temp [TODO] TODO - # @param op_mode [TODO] TODO - # @param setpoint_schedule [TODO] TODO - # @param control_setpoint_schedule [TODO] TODO + # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest + # @param amb_temp_sensor [OpenStudio::Model::EnergyManagementSystemSensor] HPWH ambient temperature sensor + # @param hpwh_top_element_sp [OpenStudio::Model::ScheduleConstant] HPWH top element setpoint schedule + # @param hpwh_bottom_element_sp [OpenStudio::Model::ScheduleConstant] HPWH bottom element setpoint schedule + # @param min_temp [Double] Minimum temperature for compressor operation (F) + # @param max_temp [Double] Maximum temperature for compressor operation (F) + # @param sensted_setpoint_schedule [OpenStudio::Model::ScheduleConstant or OpenStudio::Model::ScheduleRuleset] Setpoint temperature schedule (sensed) + # @param control_setpoint_schedule [OpenStudio::Model::ScheduleConstant or OpenStudio::Model::ScheduleRuleset] Setpoint temperature schedule (controlled) # @param schedules_file [SchedulesFile] SchedulesFile wrapper class instance of detailed schedule files - # @return [TODO] TODO - def self.add_hpwh_control_program(model, runner, obj_name, amb_temp_sensor, hpwh_top_element_sp, hpwh_bottom_element_sp, min_temp, max_temp, op_mode, setpoint_schedule, control_setpoint_schedule, schedules_file) + # @return [OpenStudio::Model::EnergyManagementSystemProgram] The HPWH control program + def self.apply_hpwh_control_program(model, runner, obj_name, water_heating_system, amb_temp_sensor, hpwh_top_element_sp, hpwh_bottom_element_sp, min_temp, max_temp, sensted_setpoint_schedule, control_setpoint_schedule, schedules_file) # Lower element is enabled if the ambient air temperature prevents the HP from running leschedoverride_actuator = Model.add_ems_actuator( name: "#{obj_name} LESchedOverride", @@ -1341,7 +1342,7 @@ def self.add_hpwh_control_program(model, runner, obj_name, amb_temp_sensor, hpwh model, name: "#{obj_name} T_set", output_var_or_meter_name: 'Schedule Value', - key_name: setpoint_schedule.name + key_name: sensted_setpoint_schedule.name ) op_mode_schedule = nil @@ -1358,7 +1359,9 @@ def self.add_hpwh_control_program(model, runner, obj_name, amb_temp_sensor, hpwh key_name: op_mode_schedule.name ) - runner.registerWarning("Both '#{SchedulesFile::Columns[:WaterHeaterOperatingMode].name}' schedule file and operating mode provided; the latter will be ignored.") if !op_mode.nil? + if not water_heating_system.operating_mode.nil? + runner.registerWarning("Both '#{SchedulesFile::Columns[:WaterHeaterOperatingMode].name}' schedule file and operating mode provided; the latter will be ignored.") + end end t_offset = 9.0 # C @@ -1371,42 +1374,42 @@ def self.add_hpwh_control_program(model, runner, obj_name, amb_temp_sensor, hpwh ) hpwh_ctrl_program.addLine("Set #{hpwhschedoverride_actuator.name} = #{t_set_sensor.name}") # If in HP only mode: still enable elements if ambient temperature is out of bounds, otherwise disable elements - if op_mode == HPXML::WaterHeaterOperatingModeHeatPumpOnly + if water_heating_system.operating_mode == HPXML::WaterHeaterOperatingModeHeatPumpOnly hpwh_ctrl_program.addLine("If (#{amb_temp_sensor.name}<#{min_temp_c}) || (#{amb_temp_sensor.name}>#{max_temp_c})") - hpwh_ctrl_program.addLine("Set #{leschedoverride_actuator.name} = #{t_set_sensor.name}") - hpwh_ctrl_program.addLine("Set #{ueschedoverride_actuator.name} = #{t_set_sensor.name}") + hpwh_ctrl_program.addLine(" Set #{leschedoverride_actuator.name} = #{t_set_sensor.name}") + hpwh_ctrl_program.addLine(" Set #{ueschedoverride_actuator.name} = #{t_set_sensor.name}") hpwh_ctrl_program.addLine('Else') - hpwh_ctrl_program.addLine("Set #{leschedoverride_actuator.name} = 0") - hpwh_ctrl_program.addLine("Set #{ueschedoverride_actuator.name} = 0") + hpwh_ctrl_program.addLine(" Set #{leschedoverride_actuator.name} = 0") + hpwh_ctrl_program.addLine(" Set #{ueschedoverride_actuator.name} = 0") hpwh_ctrl_program.addLine('EndIf') else # First, check if ambient temperature is out of bounds for HP operation, if so enable lower element hpwh_ctrl_program.addLine("If (#{amb_temp_sensor.name}<#{min_temp_c}) || (#{amb_temp_sensor.name}>#{max_temp_c})") - hpwh_ctrl_program.addLine("Set #{ueschedoverride_actuator.name} = #{t_set_sensor.name}") - hpwh_ctrl_program.addLine("Set #{leschedoverride_actuator.name} = #{t_set_sensor.name}") + hpwh_ctrl_program.addLine(" Set #{ueschedoverride_actuator.name} = #{t_set_sensor.name}") + hpwh_ctrl_program.addLine(" Set #{leschedoverride_actuator.name} = #{t_set_sensor.name}") hpwh_ctrl_program.addLine('Else') - hpwh_ctrl_program.addLine("Set #{ueschedoverride_actuator.name} = #{t_set_sensor.name} - #{t_offset}") - hpwh_ctrl_program.addLine("Set #{leschedoverride_actuator.name} = 0") + hpwh_ctrl_program.addLine(" Set #{ueschedoverride_actuator.name} = #{t_set_sensor.name} - #{t_offset}") + hpwh_ctrl_program.addLine(" Set #{leschedoverride_actuator.name} = 0") hpwh_ctrl_program.addLine('EndIf') # Scheduled operating mode: if in HP only mode, disable both elements (this will override prior logic) if not op_mode_schedule.nil? hpwh_ctrl_program.addLine("If #{op_mode_sensor.name} == 1") - hpwh_ctrl_program.addLine("Set #{ueschedoverride_actuator.name} = 0") + hpwh_ctrl_program.addLine(" Set #{ueschedoverride_actuator.name} = 0") hpwh_ctrl_program.addLine('Else') - hpwh_ctrl_program.addLine("Set #{ueschedoverride_actuator.name} = #{t_set_sensor.name} - #{t_offset}") + hpwh_ctrl_program.addLine(" Set #{ueschedoverride_actuator.name} = #{t_set_sensor.name} - #{t_offset}") hpwh_ctrl_program.addLine('EndIf') end end return hpwh_ctrl_program end - # TODO + # Sets the tank losses for a stratified water heater tank. # - # @param tank [TODO] TODO - # @param u_tank [TODO] TODO + # @param tank [OpenStudio::Model::WaterHeaterStratified] The stratified tank + # @param u_tank [Double] The heat loss U-factor (Btu/hr-ft^2-F) # @param unit_multiplier [Integer] Number of similar dwelling units # @return [nil] - def self.set_stratified_tank_ua(tank, u_tank, unit_multiplier) + def self.apply_stratified_tank_losses(tank, u_tank, unit_multiplier) node_ua = [0] * 12 # Max number of nodes in E+ stratified tank model if unit_multiplier == 1 tank.setUniformSkinLossCoefficientperUnitAreatoAmbientTemperature(u_tank) @@ -1448,11 +1451,11 @@ def self.set_stratified_tank_ua(tank, u_tank, unit_multiplier) tank.setNode12AdditionalLossCoefficient(node_ua[11]) end - # TODO + # Gets the combination boiler and plant loop OpenStudio objects associated with the HPXML water heating system. # # @param model [OpenStudio::Model::Model] OpenStudio Model object - # @param heating_source_id [TODO] TODO - # @return [TODO] TODO + # @param heating_source_id [String] ID of the HPXML water heating system + # @return [Array] Boiler object, PlantLoop object def self.get_combi_boiler_and_plant_loop(model, heating_source_id) # Search for the right boiler OS object boiler_hw = nil @@ -1478,38 +1481,34 @@ def self.get_combi_boiler_and_plant_loop(model, heating_source_id) return boiler_hw, plant_loop_hw end - # TODO + # Adds a desuperheater for the given HPXML water heating system to the OpenStudio model. # - # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest # @param model [OpenStudio::Model::Model] OpenStudio Model object - # @return [TODO] TODO - def self.get_desuperheatercoil(water_heating_system, model) + # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings + # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest + # @param tank [OpenStudio::Model::WaterHeaterMixed or OpenStudio::Model::WaterHeaterStratified] The water heater tank + # @param loc_space [OpenStudio::Model::Space] The space where the water heater is located + # @param loc_schedule [OpenStudio::Model::ScheduleConstant] The temperature schedule, if not located in a space + # @param plant_loop [OpenStudio::Model::PlantLoop] The DHW plant loop + # @param unit_multiplier [Integer] Number of similar dwelling units + # @return [nil] + def self.apply_desuperheater(model, runner, water_heating_system, tank, loc_space, loc_schedule, plant_loop, unit_multiplier) + return unless water_heating_system.uses_desuperheater + + # Get the HVAC cooling coil for the desuperheater + desuperheater_clg_coil = nil (model.getCoilCoolingDXSingleSpeeds + model.getCoilCoolingDXMultiSpeeds + model.getCoilCoolingWaterToAirHeatPumpEquationFits).each do |clg_coil| sys_id = clg_coil.additionalProperties.getFeatureAsString('HPXML_ID') if sys_id.is_initialized && sys_id.get == water_heating_system.related_hvac_idref - return clg_coil + desuperheater_clg_coil = clg_coil end end - fail "RelatedHVACSystem '#{water_heating_system.related_hvac_idref}' for water heating system '#{water_heating_system.id}' is not currently supported for desuperheaters." - end - - # TODO - # - # @param model [OpenStudio::Model::Model] OpenStudio Model object - # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings - # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest - # @param tank [TODO] TODO - # @param loc_space [OpenStudio::Model::Space] The space where the water heater is located - # @param loc_schedule [OpenStudio::Model::ScheduleConstant] The temperature schedule for where the water heater is located, if not in a space - # @param loop [TODO] TODO - # @param unit_multiplier [Integer] Number of similar dwelling units - # @return [TODO] TODO - def self.add_desuperheater(model, runner, water_heating_system, tank, loc_space, loc_schedule, loop, unit_multiplier) - return unless water_heating_system.uses_desuperheater + if desuperheater_clg_coil.nil? + fail "RelatedHVACSystem '#{water_heating_system.related_hvac_idref}' for water heating system '#{water_heating_system.id}' is not currently supported for desuperheaters." + end - desuperheater_clg_coil = get_desuperheatercoil(water_heating_system, model) reclaimed_efficiency = 0.25 # default desuperheater_name = "#{tank.name} desuperheater" @@ -1518,25 +1517,26 @@ def self.add_desuperheater(model, runner, water_heating_system, tank, loc_space, storage_vol_actual = calc_storage_tank_actual_vol(vol, nil) assumed_ua = 6.0 # Btu/hr-F, tank ua calculated based on 1.0 standby_loss and 50gal nominal vol storage_tank_name = "#{tank.name} storage tank" - # reduce tank setpoint to enable desuperheater setpoint at t_set + if water_heating_system.temperature.nil? fail "Detailed setpoints for water heating system '#{water_heating_system.id}' is not currently supported for desuperheaters." else - tank_setpoint = get_t_set_c(water_heating_system.temperature - 5.0, HPXML::WaterHeaterTypeStorage) + # reduce tank setpoint to enable desuperheater setpoint at t_set + t_set_c = get_t_set_c(water_heating_system.temperature - 5.0, HPXML::WaterHeaterTypeStorage) end - storage_tank = create_new_heater(name: storage_tank_name, - act_vol: storage_vol_actual, - t_set_c: tank_setpoint, - loc_space: loc_space, - loc_schedule: loc_schedule, - model: model, - runner: runner, - ua: assumed_ua, - is_dsh_storage: true, - unit_multiplier: unit_multiplier) - - loop.addSupplyBranchForComponent(storage_tank) + storage_tank = apply_water_heater(name: storage_tank_name, + act_vol: storage_vol_actual, + t_set_c: t_set_c, + loc_space: loc_space, + loc_schedule: loc_schedule, + model: model, + runner: runner, + ua: assumed_ua, + is_dsh_storage: true, + unit_multiplier: unit_multiplier) + + plant_loop.addSupplyBranchForComponent(storage_tank) tank.addToNode(storage_tank.supplyOutletModelObject.get.to_Node.get) # Create a schedule for desuperheater @@ -1549,50 +1549,53 @@ def self.add_desuperheater(model, runner, water_heating_system, tank, loc_space, limits: EPlus::ScheduleTypeLimitsTemperature ) - # create a desuperheater object + # Create desuperheater object desuperheater = OpenStudio::Model::CoilWaterHeatingDesuperheater.new(model, new_schedule) desuperheater.setName(desuperheater_name) desuperheater.setMaximumInletWaterTemperatureforHeatReclaim(100) desuperheater.setDeadBandTemperatureDifference(0.2) desuperheater.setRatedHeatReclaimRecoveryEfficiency(reclaimed_efficiency) desuperheater.addToHeatRejectionTarget(storage_tank) - # FUTURE: Desuperheater pump power? - desuperheater.setWaterPumpPower(0) - # attach to the clg coil source + desuperheater.setWaterPumpPower(0) # FUTURE: Desuperheater pump power? desuperheater.setHeatingSource(desuperheater_clg_coil) desuperheater.setWaterFlowRate(0.0001 * unit_multiplier) desuperheater.additionalProperties.setFeature('HPXML_ID', water_heating_system.id) # Used by reporting measure end - # TODO + # Calculates an Energy Factor (EF) from a Uniform Energy Factor (UEF). + # + # Source: RESNET's Interpretation on Water Heater UEF + # https://www.resnet.us/wp-content/uploads/Interpretation-301-2014-012-Water-Heater-UEF.pdf + # Note that this is a regression based on products on the market, not a conversion. # # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest - # @return [TODO] TODO + # @return [Double] The Energy Factor def self.calc_ef_from_uef(water_heating_system) - # Interpretation on Water Heater UEF if water_heating_system.fuel_type == HPXML::FuelTypeElectricity - if water_heating_system.water_heater_type == HPXML::WaterHeaterTypeStorage + case water_heating_system.water_heater_type + when HPXML::WaterHeaterTypeStorage return [2.4029 * water_heating_system.uniform_energy_factor - 1.2844, 0.96].min - elsif water_heating_system.water_heater_type == HPXML::WaterHeaterTypeTankless + when HPXML::WaterHeaterTypeTankless return water_heating_system.uniform_energy_factor - elsif water_heating_system.water_heater_type == HPXML::WaterHeaterTypeHeatPump + when HPXML::WaterHeaterTypeHeatPump return 1.2101 * water_heating_system.uniform_energy_factor - 0.6052 end else # Fuel - if water_heating_system.water_heater_type == HPXML::WaterHeaterTypeStorage + case water_heating_system.water_heater_type + when HPXML::WaterHeaterTypeStorage return 0.9066 * water_heating_system.uniform_energy_factor + 0.0711 - elsif water_heating_system.water_heater_type == HPXML::WaterHeaterTypeTankless + when HPXML::WaterHeaterTypeTankless return water_heating_system.uniform_energy_factor end end fail 'Unexpected water heater.' end - # TODO + # Calculates water heater storage tank surface areas. # - # @param act_vol [TODO] TODO - # @param height [TODO] TODO - # @return [TODO] TODO + # @param act_vol [Double] Actual tank volume (gal) + # @param height [Double] Tank height (ft) + # @return [Array] Tank total surface area (ft), Tank side surface area (ft) def self.calc_tank_areas(act_vol, height = nil) if height.nil? height = DefaultTankHeight @@ -1600,20 +1603,20 @@ def self.calc_tank_areas(act_vol, height = nil) diameter = 2.0 * (UnitConversions.convert(act_vol, 'gal', 'ft^3') / (height * Math::PI))**0.5 # feet a_top = Math::PI * diameter**2.0 / 4.0 # sqft a_side = Math::PI * diameter * height # sqft - surface_area = 2.0 * a_top + a_side # sqft + a_total = 2.0 * a_top + a_side # sqft - return surface_area, a_side + return a_total, a_side end - # TODO + # Calculates tank losses for the combination boiler given its standby loss value. # - # @param act_vol [TODO] TODO + # @param act_vol [Double] Actual tank volume (gal) # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest - # @param a_side [TODO] TODO - # @param solar_fraction [TODO] TODO + # @param a_side [Double] Tank side surface area (ft^3) + # @param solar_fraction [Double] Portion of hot water load served by an attached solar thermal system # @param nbeds [Integer] Number of bedrooms in the dwelling unit - # @return [TODO] TODO - def self.calc_indirect_ua_with_standbyloss(act_vol, water_heating_system, a_side, solar_fraction, nbeds = nil) + # @return [Double] Tank loss UA factor (Btu/hr-F) + def self.calc_combi_tank_losses(act_vol, water_heating_system, a_side, solar_fraction, nbeds = nil) standby_loss_units = water_heating_system.standby_loss_units standby_loss_value = water_heating_system.standby_loss_value @@ -1641,17 +1644,18 @@ def self.calc_indirect_ua_with_standbyloss(act_vol, water_heating_system, a_side return ua end - # TODO + # Adds an EMS program to increase/decrease the energy consumption of the water heater based on + # the energy consumption adjustment factor (EC_adj). # # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param hpxml_bldg [HPXML::Building] HPXML Building object representing an individual dwelling unit - # @param heater [TODO] TODO + # @param water_heater [OpenStudio::Model::WaterHeaterMixed or OpenStudio::Model::WaterHeaterStratified or OpenStudio::Model::WaterHeaterHeatPumpWrappedCondenser] The water heater object # @param loc_space [OpenStudio::Model::Space] The space where the water heater is located # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest # @param unit_multiplier [Integer] Number of similar dwelling units - # @param combi_boiler [TODO] TODO - # @return [TODO] TODO - def self.add_ec_adj(model, hpxml_bldg, heater, loc_space, water_heating_system, unit_multiplier, combi_boiler = nil) + # @param combi_boiler [OpenStudio::Model::BoilerHotWater] The boiler object if the HPXML water heating system is a combi boiler + # @return [nil] + def self.apply_ec_adj_program(model, hpxml_bldg, water_heater, loc_space, water_heating_system, unit_multiplier, combi_boiler = nil) ec_adj = get_dist_energy_consumption_adjustment(hpxml_bldg, water_heating_system) adjustment = ec_adj - 1.0 @@ -1660,9 +1664,9 @@ def self.add_ec_adj(model, hpxml_bldg, heater, loc_space, water_heating_system, end if water_heating_system.water_heater_type == HPXML::WaterHeaterTypeHeatPump - tank = heater.tank + tank = water_heater.tank else - tank = heater + tank = water_heater end if [HPXML::WaterHeaterTypeCombiStorage, HPXML::WaterHeaterTypeCombiTankless].include? water_heating_system.water_heater_type fuel_type = water_heating_system.related_hvac_system.heating_system_fuel @@ -1706,15 +1710,15 @@ def self.add_ec_adj(model, hpxml_bldg, heater, loc_space, water_heating_system, if water_heating_system.water_heater_type == HPXML::WaterHeaterTypeHeatPump ec_adj_hp_sensor = Model.add_ems_sensor( model, - name: "#{heater.dXCoil.name} energy", + name: "#{water_heater.dXCoil.name} energy", output_var_or_meter_name: "Cooling Coil Water Heating #{EPlus::FuelTypeElectricity} Rate", - key_name: heater.dXCoil.name + key_name: water_heater.dXCoil.name ) ec_adj_fan_sensor = Model.add_ems_sensor( model, - name: "#{heater.fan.name} energy", + name: "#{water_heater.fan.name} energy", output_var_or_meter_name: "Fan #{EPlus::FuelTypeElectricity} Rate", - key_name: heater.fan.name + key_name: water_heater.fan.name ) end end @@ -1734,7 +1738,7 @@ def self.add_ec_adj(model, hpxml_bldg, heater, loc_space, water_heating_system, # Actuators ec_adj_actuator = Model.add_ems_actuator( - name: "#{heater.name} ec adj act", + name: "#{water_heater.name} ec adj act", model_object: ec_adj_object, comp_type_and_control: EPlus::EMSActuatorOtherEquipmentPower ) @@ -1742,7 +1746,7 @@ def self.add_ec_adj(model, hpxml_bldg, heater, loc_space, water_heating_system, # Program ec_adj_program = Model.add_ems_program( model, - name: "#{heater.name} EC_adj" + name: "#{water_heater.name} EC_adj" ) ec_adj_program.addLine('If WarmupFlag == 0') # Prevent a non-zero adjustment in the first hour because of the warmup period if [HPXML::WaterHeaterTypeCombiStorage, HPXML::WaterHeaterTypeCombiTankless].include? water_heating_system.water_heater_type @@ -1763,31 +1767,30 @@ def self.add_ec_adj(model, hpxml_bldg, heater, loc_space, water_heating_system, # Program Calling Manager Model.add_ems_program_calling_manager( model, - name: "#{heater.name} EC_adj ProgramManager", + name: "#{water_heater.name} EC_adj ProgramManager", calling_point: 'EndOfSystemTimestepBeforeHVACReporting', ems_programs: [ec_adj_program] ) end - # TODO + # Gets the water heater setpoint temperature deadband. # - # @param wh_type [TODO] TODO - # @return [TODO] TODO - def self.deadband(wh_type) + # @param wh_type [String] Type of water heater (HPXML::WaterHeaterTypeXXX) + # @return [Double] Temperature deadband (C) + def self.get_deadband(wh_type) if [HPXML::WaterHeaterTypeStorage, HPXML::WaterHeaterTypeCombiStorage].include? wh_type - return 2.0 # C + return 2.0 else - return 0.0 # C + return 0.0 end end - # TODO + # Calculates the water heater actual volume from its nominal/rated volume. # - # @param vol [TODO] TODO - # @param fuel [TODO] TODO - # @return [TODO] TODO + # @param vol [Double] Nominal tank volume (gal) + # @param fuel [String] Water heater fuel type (HPXML::FuelTypeXXX) + # @return [Double] Actual tank volume (gal) def self.calc_storage_tank_actual_vol(vol, fuel) - # Convert the nominal tank volume to an actual volume if fuel.nil? act_vol = 0.95 * vol # indirect tank else @@ -1800,22 +1803,21 @@ def self.calc_storage_tank_actual_vol(vol, fuel) return act_vol end - # TODO + # Disaggregates the water heater's (uniform) energy factor into tank losses and burner efficiency. + # + # If using EF: + # Calculations based on the Energy Factor and Recovery Efficiency of the tank + # Source: Burch and Erickson 2004 - http://www.nrel.gov/docs/gen/fy04/36035.pdf + # IF using UEF: + # Calculations based on the Uniform Energy Factor, First Hour Rating, and Recovery Efficiency of the tank + # Source: Maguire and Roberts 2020 - https://www.ashrae.org/file%20library/conferences/specialty%20conferences/2020%20building%20performance/papers/d-bsc20-c039.pdf # - # @param act_vol [TODO] TODO + # @param act_vol [Double] Actual tank volume (gal) # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest - # @param solar_fraction [TODO] TODO + # @param solar_fraction [Double] Portion of hot water load served by an attached solar thermal system # @param nbeds [Integer] Number of bedrooms in the dwelling unit - # @return [TODO] TODO - def self.calc_tank_UA(act_vol, water_heating_system, solar_fraction, nbeds) - # If using EF: - # Calculates the U value, UA of the tank and conversion efficiency (eta_c) - # based on the Energy Factor and recovery efficiency of the tank - # Source: Burch and Erickson 2004 - http://www.nrel.gov/docs/gen/fy04/36035.pdf - # IF using UEF: - # Calculates the U value, UA of the tank and conversion efficiency (eta_c) - # based on the Uniform Energy Factor, First Hour Rating, and Recovery Efficiency of the tank - # Source: Maguire and Roberts 2020 - https://www.ashrae.org/file%20library/conferences/specialty%20conferences/2020%20building%20performance/papers/d-bsc20-c039.pdf + # @return [Array] Tank loss U-factor (Btu/hr-ft^2-F), tank loss UA factor (Btu/hr-F), burner efficiency (frac) + def self.disaggregate_tank_losses_and_burner_efficiency(act_vol, water_heating_system, solar_fraction, nbeds) if water_heating_system.water_heater_type == HPXML::WaterHeaterTypeTankless if not water_heating_system.energy_factor.nil? eta_c = water_heating_system.energy_factor * water_heating_system.performance_adjustment @@ -1835,13 +1837,14 @@ def self.calc_tank_UA(act_vol, water_heating_system, solar_fraction, nbeds) volume_drawn = 64.3 # gal/day elsif not water_heating_system.uniform_energy_factor.nil? t = 125.0 # F - if water_heating_system.usage_bin == HPXML::WaterHeaterUsageBinVerySmall + case water_heating_system.usage_bin + when HPXML::WaterHeaterUsageBinVerySmall volume_drawn = 10.0 # gal - elsif water_heating_system.usage_bin == HPXML::WaterHeaterUsageBinLow + when HPXML::WaterHeaterUsageBinLow volume_drawn = 38.0 # gal - elsif water_heating_system.usage_bin == HPXML::WaterHeaterUsageBinMedium + when HPXML::WaterHeaterUsageBinMedium volume_drawn = 55.0 # gal - elsif water_heating_system.usage_bin == HPXML::WaterHeaterUsageBinHigh + when HPXML::WaterHeaterUsageBinHigh volume_drawn = 84.0 # gal end end @@ -1881,13 +1884,14 @@ def self.calc_tank_UA(act_vol, water_heating_system, solar_fraction, nbeds) return u, ua, eta_c end - # TODO + # Calculates an adjustment to the tank loss UA factor to account for the presence + # of a water heater jacket (insulation). # # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest - # @param ua_pre [TODO] TODO - # @param a_side [TODO] TODO - # @return [TODO] TODO - def self.apply_tank_jacket(water_heating_system, ua_pre, a_side) + # @param ua [Double] Tank loss UA factor (Btu/hr-F) + # @param a_side [Double] Tank side surface area (ft^3) + # @return [Double] Adjusted tank loss UA factor (Btu/hr-F) + def self.apply_tank_jacket(water_heating_system, ua, a_side) if not water_heating_system.jacket_r_value.nil? skin_insulation_R = 5.0 # R5 if water_heating_system.fuel_type.nil? # indirect water heater, etc. Assume 2 inch skin insulation @@ -1898,58 +1902,59 @@ def self.apply_tank_jacket(water_heating_system, ua_pre, a_side) ef = calc_ef_from_uef(water_heating_system) end if ef < 0.7 - skin_insulation_t = 1.0 # inch + skin_insulation_t = 1.0 # inch, assumed else - skin_insulation_t = 2.0 # inch + skin_insulation_t = 2.0 # inch, assumed end else # electric - skin_insulation_t = 2.0 # inch + skin_insulation_t = 2.0 # inch, assumed end # water heater wrap calculation based on: # Modeling Water Heat Wraps in BEopt DRAFT Technical Note # Authors: Ben Polly and Jay Burch (NREL) u_pre_skin = 1.0 / (skin_insulation_t * skin_insulation_R + 1.0 / 1.3 + 1.0 / 52.8) # Btu/hr-ft^2-F = (1 / hout + kins / tins + t / hin)^-1 - ua = ua_pre - water_heating_system.jacket_r_value / (1.0 / u_pre_skin + water_heating_system.jacket_r_value) * u_pre_skin * a_side + ua_adj = ua - water_heating_system.jacket_r_value / (1.0 / u_pre_skin + water_heating_system.jacket_r_value) * u_pre_skin * a_side else - ua = ua_pre + ua_adj = ua end - return ua + return ua_adj end - # TODO + # Calculates an adjustment to the tank loss UA factor for a shared water heater, in which we + # apportion the tank losses to the dwelling unit. # # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest - # @param ua [TODO] TODO + # @param ua [Double] Tank loss UA factor (Btu/hr-F) # @param nbeds [Integer] Number of bedrooms in the dwelling unit - # @return [TODO] TODO + # @return [Double] Adjusted tank loss UA factor (Btu/hr-F) def self.apply_shared_adjustment(water_heating_system, ua, nbeds) if water_heating_system.is_shared_system - # Apportion shared water heater energy use due to tank losses to the dwelling unit - ua = ua * [nbeds.to_f, 1.0].max / water_heating_system.number_of_bedrooms_served.to_f + return ua * [nbeds.to_f, 1.0].max / water_heating_system.number_of_bedrooms_served.to_f + else + return ua end - return ua end - # TODO + # Adds a water heater object to the OpenStudio model. # - # @param name [TODO] TODO + # @param name [String] Name for the OpenStudio object # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest - # @param act_vol [TODO] TODO - # @param t_set_c [TODO] TODO + # @param act_vol [Double] Actual tank volume (gal) + # @param t_set_c [Double] Water heater setpoint including deadband (C) # @param loc_space [OpenStudio::Model::Space] The space where the water heater is located - # @param loc_schedule [OpenStudio::Model::ScheduleConstant] The temperature schedule for where the water heater is located, if not in a space + # @param loc_schedule [OpenStudio::Model::ScheduleConstant] The temperature schedule, if not located in a space # @param model [OpenStudio::Model::Model] OpenStudio Model object # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings - # @param u [TODO] TODO - # @param ua [TODO] TODO - # @param eta_c [TODO] TODO - # @param is_dsh_storage [TODO] TODO - # @param is_combi [TODO] TODO + # @param u [Double] Tank loss coefficient (FIXME) + # @param ua [Double] Tank loss UA factor (Btu/hr-F) + # @param eta_c [Double] Burner efficiency (frac) + # @param is_dsh_storage [Boolean] True if this is a desuperheater storage tank + # @param is_combi [Boolean] True if this is a combination boiler # @param schedules_file [SchedulesFile] SchedulesFile wrapper class instance of detailed schedule files # @param unavailable_periods [HPXML::UnavailablePeriods] Object that defines periods for, e.g., power outages or vacancies # @param unit_multiplier [Integer] Number of similar dwelling units - # @return [TODO] TODO - def self.create_new_heater(name:, water_heating_system: nil, act_vol:, t_set_c: nil, loc_space:, loc_schedule: nil, model:, runner:, u: nil, ua:, eta_c: nil, is_dsh_storage: false, is_combi: false, schedules_file: nil, unavailable_periods: [], unit_multiplier: 1.0) + # @return [OpenStudio::Model::WaterHeaterMixed or OpenStudio::Model::WaterHeaterStratified] Water heater object + def self.apply_water_heater(name:, water_heating_system: nil, act_vol:, t_set_c: nil, loc_space:, loc_schedule: nil, model:, runner:, u: nil, ua:, eta_c: nil, is_dsh_storage: false, is_combi: false, schedules_file: nil, unavailable_periods: [], unit_multiplier: 1.0) # storage tank doesn't require water_heating_system class argument being passed if is_dsh_storage || is_combi fuel = nil @@ -1974,44 +1979,44 @@ def self.create_new_heater(name:, water_heating_system: nil, act_vol:, t_set_c: h_tank = UnitConversions.convert(DefaultTankHeight, 'ft', 'm') # Add a WaterHeater:Stratified to the model - new_heater = OpenStudio::Model::WaterHeaterStratified.new(model) - new_heater.setEndUseSubcategory('Domestic Hot Water') - new_heater.setTankVolume(UnitConversions.convert(act_vol, 'gal', 'm^3')) - new_heater.setTankHeight(h_tank) - new_heater.setMaximumTemperatureLimit(90) - new_heater.setHeaterPriorityControl('MasterSlave') - new_heater.setHeater1Capacity(UnitConversions.convert(cap, 'kBtu/hr', 'W')) - new_heater.setHeater1Height((1.0 - (4 - 0.5) / 15) * h_tank) # in the 4th node of a 15-node tank (counting from top); height of upper element based on TRNSYS assumptions for an ERWH - new_heater.setHeater1DeadbandTemperatureDifference(5.556) - new_heater.setHeater2Capacity(UnitConversions.convert(cap, 'kBtu/hr', 'W')) - new_heater.setHeater2Height((1.0 - (13 - 0.5) / 15) * h_tank) # in the 13th node of a 15-node tank (counting from top); height of upper element based on TRNSYS assumptions for an ERWH - new_heater.setHeater2DeadbandTemperatureDifference(5.556) - new_heater.setHeaterThermalEfficiency(1.0) - new_heater.setNumberofNodes(12) - new_heater.setAdditionalDestratificationConductivity(0) - new_heater.setUseSideDesignFlowRate(UnitConversions.convert(act_vol, 'gal', 'm^3') / 60.1) - new_heater.setSourceSideDesignFlowRate(0) - new_heater.setSourceSideFlowControlMode('') - new_heater.setSourceSideInletHeight((1.0 - (1 - 0.5) / 15) * h_tank) # in the 1st node of a 15-node tank (counting from top) - new_heater.setSourceSideOutletHeight((1.0 - (15 - 0.5) / 15) * h_tank) # in the 15th node of a 15-node tank (counting from top) - new_heater.setSkinLossFractiontoZone(1.0 / unit_multiplier) # Tank losses are multiplied by E+ zone multiplier, so need to compensate here - new_heater.setOffCycleFlueLossFractiontoZone(1.0 / unit_multiplier) - set_stratified_tank_ua(new_heater, u, unit_multiplier) + water_heater = OpenStudio::Model::WaterHeaterStratified.new(model) + water_heater.setEndUseSubcategory('Domestic Hot Water') + water_heater.setTankVolume(UnitConversions.convert(act_vol, 'gal', 'm^3')) + water_heater.setTankHeight(h_tank) + water_heater.setMaximumTemperatureLimit(90) + water_heater.setHeaterPriorityControl('MasterSlave') + water_heater.setHeater1Capacity(UnitConversions.convert(cap, 'kBtu/hr', 'W')) + water_heater.setHeater1Height((1.0 - (4 - 0.5) / 15) * h_tank) # in the 4th node of a 15-node tank (counting from top); height of upper element based on TRNSYS assumptions for an ERWH + water_heater.setHeater1DeadbandTemperatureDifference(5.556) + water_heater.setHeater2Capacity(UnitConversions.convert(cap, 'kBtu/hr', 'W')) + water_heater.setHeater2Height((1.0 - (13 - 0.5) / 15) * h_tank) # in the 13th node of a 15-node tank (counting from top); height of upper element based on TRNSYS assumptions for an ERWH + water_heater.setHeater2DeadbandTemperatureDifference(5.556) + water_heater.setHeaterThermalEfficiency(1.0) + water_heater.setNumberofNodes(12) + water_heater.setAdditionalDestratificationConductivity(0) + water_heater.setUseSideDesignFlowRate(UnitConversions.convert(act_vol, 'gal', 'm^3') / 60.1) + water_heater.setSourceSideDesignFlowRate(0) + water_heater.setSourceSideFlowControlMode('') + water_heater.setSourceSideInletHeight((1.0 - (1 - 0.5) / 15) * h_tank) # in the 1st node of a 15-node tank (counting from top) + water_heater.setSourceSideOutletHeight((1.0 - (15 - 0.5) / 15) * h_tank) # in the 15th node of a 15-node tank (counting from top) + water_heater.setSkinLossFractiontoZone(1.0 / unit_multiplier) # Tank losses are multiplied by E+ zone multiplier, so need to compensate here + water_heater.setOffCycleFlueLossFractiontoZone(1.0 / unit_multiplier) + apply_stratified_tank_losses(water_heater, u, unit_multiplier) else - new_heater = OpenStudio::Model::WaterHeaterMixed.new(model) - new_heater.setTankVolume(UnitConversions.convert(act_vol, 'gal', 'm^3')) - new_heater.setHeaterThermalEfficiency(eta_c) unless eta_c.nil? - new_heater.setMaximumTemperatureLimit(99.0) + water_heater = OpenStudio::Model::WaterHeaterMixed.new(model) + water_heater.setTankVolume(UnitConversions.convert(act_vol, 'gal', 'm^3')) + water_heater.setHeaterThermalEfficiency(eta_c) unless eta_c.nil? + water_heater.setMaximumTemperatureLimit(99.0) if [HPXML::WaterHeaterTypeTankless, HPXML::WaterHeaterTypeCombiTankless].include? tank_type - new_heater.setHeaterControlType('Modulate') + water_heater.setHeaterControlType('Modulate') else - new_heater.setHeaterControlType('Cycle') + water_heater.setHeaterControlType('Cycle') end - new_heater.setDeadbandTemperatureDifference(deadband(tank_type)) + water_heater.setDeadbandTemperatureDifference(get_deadband(tank_type)) # Capacity, storage tank to be 0 - new_heater.setHeaterMaximumCapacity(UnitConversions.convert(cap, 'kBtu/hr', 'W')) - new_heater.setHeaterMinimumCapacity(0.0) + water_heater.setHeaterMaximumCapacity(UnitConversions.convert(cap, 'kBtu/hr', 'W')) + water_heater.setHeaterMinimumCapacity(0.0) # Set fraction of heat loss from tank to ambient (vs out flue) # Based on lab testing done by LBNL @@ -2032,68 +2037,68 @@ def self.create_new_heater(name:, water_heating_system: nil, act_vol:, t_set_c: skinlossfrac = 0.96 # Condensing end end - new_heater.setOffCycleLossFractiontoThermalZone(skinlossfrac / unit_multiplier) # Tank losses are multiplied by E+ zone multiplier, so need to compensate here - new_heater.setOnCycleLossFractiontoThermalZone(1.0 / unit_multiplier) # Tank losses are multiplied by E+ zone multiplier, so need to compensate here + water_heater.setOffCycleLossFractiontoThermalZone(skinlossfrac / unit_multiplier) # Tank losses are multiplied by E+ zone multiplier, so need to compensate here + water_heater.setOnCycleLossFractiontoThermalZone(1.0 / unit_multiplier) # Tank losses are multiplied by E+ zone multiplier, so need to compensate here ua_w_k = UnitConversions.convert(ua, 'Btu/(hr*F)', 'W/K') - new_heater.setOnCycleLossCoefficienttoAmbientTemperature(ua_w_k) - new_heater.setOffCycleLossCoefficienttoAmbientTemperature(ua_w_k) + water_heater.setOnCycleLossCoefficienttoAmbientTemperature(ua_w_k) + water_heater.setOffCycleLossCoefficienttoAmbientTemperature(ua_w_k) end - assign_water_heater_setpoint(runner, model, new_heater, schedules_file, t_set_c, unavailable_periods) + apply_setpoint(runner, model, water_heater, schedules_file, t_set_c, unavailable_periods) if not water_heating_system.nil? - new_heater.additionalProperties.setFeature('HPXML_ID', water_heating_system.id) # Used by reporting measure + water_heater.additionalProperties.setFeature('HPXML_ID', water_heating_system.id) # Used by reporting measure end if is_combi - new_heater.additionalProperties.setFeature('IsCombiBoiler', true) # Used by reporting measure + water_heater.additionalProperties.setFeature('IsCombiBoiler', true) # Used by reporting measure end - new_heater.setName(name) - new_heater.setHeaterFuelType(EPlus.fuel_type(fuel)) unless fuel.nil? - set_wh_ambient(loc_space, loc_schedule, new_heater) + water_heater.setName(name) + water_heater.setHeaterFuelType(EPlus.fuel_type(fuel)) unless fuel.nil? + apply_ambient_temperature(loc_space, loc_schedule, water_heater) # FUTURE: These are always zero right now; develop smart defaults. - new_heater.setOffCycleParasiticFuelType(EPlus::FuelTypeElectricity) - new_heater.setOffCycleParasiticFuelConsumptionRate(0.0) - new_heater.setOffCycleParasiticHeatFractiontoTank(0) - new_heater.setOnCycleParasiticFuelType(EPlus::FuelTypeElectricity) - new_heater.setOnCycleParasiticFuelConsumptionRate(0.0) - new_heater.setOnCycleParasiticHeatFractiontoTank(0) - - return new_heater + water_heater.setOffCycleParasiticFuelType(EPlus::FuelTypeElectricity) + water_heater.setOffCycleParasiticFuelConsumptionRate(0.0) + water_heater.setOffCycleParasiticHeatFractiontoTank(0) + water_heater.setOnCycleParasiticFuelType(EPlus::FuelTypeElectricity) + water_heater.setOnCycleParasiticFuelConsumptionRate(0.0) + water_heater.setOnCycleParasiticHeatFractiontoTank(0) + + return water_heater end - # TODO + # Applies the ambient temperature conditions to the water heater. # # @param loc_space [OpenStudio::Model::Space] The space where the water heater is located - # @param loc_schedule [OpenStudio::Model::ScheduleConstant] The temperature schedule for where the water heater is located, if not in a space - # @param wh_obj [TODO] TODO + # @param loc_schedule [OpenStudio::Model::ScheduleConstant] The temperature schedule, if not located in a space + # @param water_heater [OpenStudio::Model::WaterHeaterMixed or OpenStudio::Model::WaterHeaterStratified] Water heater object # @return [nil] - def self.set_wh_ambient(loc_space, loc_schedule, wh_obj) - if wh_obj.ambientTemperatureSchedule.is_initialized - wh_obj.ambientTemperatureSchedule.get.remove + def self.apply_ambient_temperature(loc_space, loc_schedule, water_heater) + if water_heater.ambientTemperatureSchedule.is_initialized + water_heater.ambientTemperatureSchedule.get.remove end if not loc_schedule.nil? # Temperature schedule indicator - wh_obj.setAmbientTemperatureSchedule(loc_schedule) + water_heater.setAmbientTemperatureSchedule(loc_schedule) elsif not loc_space.nil? - wh_obj.setAmbientTemperatureIndicator('ThermalZone') - wh_obj.setAmbientTemperatureThermalZone(loc_space.thermalZone.get) + water_heater.setAmbientTemperatureIndicator('ThermalZone') + water_heater.setAmbientTemperatureThermalZone(loc_space.thermalZone.get) else # Located outside - wh_obj.setAmbientTemperatureIndicator('Outdoors') + water_heater.setAmbientTemperatureIndicator('Outdoors') end end - # TODO + # Applies the temperature setpoint to the water heater. # # @param runner [OpenStudio::Measure::OSRunner] Object typically used to display warnings # @param model [OpenStudio::Model::Model] OpenStudio Model object - # @param water_heater [TODO] TODO + # @param water_heater [OpenStudio::Model::WaterHeaterMixed or OpenStudio::Model::WaterHeaterStratified] Water heater object # @param schedules_file [SchedulesFile] SchedulesFile wrapper class instance of detailed schedule files - # @param t_set_c [TODO] TODO + # @param t_set_c [Double] Water heater setpoint including deadband (C) # @param unavailable_periods [HPXML::UnavailablePeriods] Object that defines periods for, e.g., power outages or vacancies # @return [nil] - def self.assign_water_heater_setpoint(runner, model, water_heater, schedules_file, t_set_c, unavailable_periods) + def self.apply_setpoint(runner, model, water_heater, schedules_file, t_set_c, unavailable_periods) setpoint_sch = nil if not schedules_file.nil? setpoint_sch = schedules_file.create_schedule_file(model, col_name: SchedulesFile::Columns[:WaterHeaterSetpoint].name) @@ -2117,51 +2122,55 @@ def self.assign_water_heater_setpoint(runner, model, water_heater, schedules_fil end end - # TODO + # Returns the water heater setpoint, accounting for any deadband, in deg-C. The deadband is currently + # centered, not single-sided; see https://github.com/NREL/OpenStudio-HPXML/issues/642. # - # @param t_set [TODO] TODO - # @param wh_type [TODO] TODO - # @return [TODO] TODO + # @param t_set [Double] Water heater setpoint (F) + # @param wh_type [String] Type of water heater (HPXML::WaterHeaterTypeXXX) + # @return [Double] Water heater setpoint including deadband (C) def self.get_t_set_c(t_set, wh_type) return if t_set.nil? - return UnitConversions.convert(t_set, 'F', 'C') + deadband(wh_type) / 2.0 # Half the deadband to account for E+ deadband + return UnitConversions.convert(t_set, 'F', 'C') + get_deadband(wh_type) / 2.0 # Half the deadband to account for E+ deadband end - # TODO + # Adds a plant loop for the water heater to the OpenStudio model. # # @param model [OpenStudio::Model::Model] OpenStudio Model object - # @param t_set_c [TODO] TODO + # @param t_set_c [Double] Water heater setpoint including deadband (C) # @param eri_version [String] Version of the ANSI/RESNET/ICC 301 Standard to use for equations/assumptions # @param unit_multiplier [Integer] Number of similar dwelling units - # @return [TODO] TODO - def self.create_new_loop(model, t_set_c, eri_version, unit_multiplier) - # Create a new plant loop for the water heater + # @return [OpenStudio::Model::PlantLoop] The plant loop + def self.add_plant_loop(model, t_set_c, eri_version, unit_multiplier) name = 'dhw loop' if t_set_c.nil? t_set_c = UnitConversions.convert(Defaults.get_water_heater_temperature(eri_version), 'F', 'C') end - loop = OpenStudio::Model::PlantLoop.new(model) - loop.setName(name) - loop.sizingPlant.setDesignLoopExitTemperature(t_set_c) - loop.sizingPlant.setLoopDesignTemperatureDifference(UnitConversions.convert(10.0, 'deltaF', 'deltaC')) - loop.setPlantLoopVolume(0.003 * unit_multiplier) # ~1 gal - loop.setMaximumLoopFlowRate(0.01 * unit_multiplier) # This size represents the physical limitations to flow due to losses in the piping system. We assume that the pipes are always adequately sized. + plant_loop = Model.add_plant_loop( + model, + name: name, + volume: 0.003 * unit_multiplier, # ~1 gal + max_flow_rate: 0.01 * unit_multiplier # This size represents the physical limitations to flow due to losses in the piping system. We assume that the pipes are always adequately sized. + ) + + sizing_plant = plant_loop.sizingPlant + sizing_plant.setDesignLoopExitTemperature(t_set_c) + sizing_plant.setLoopDesignTemperatureDifference(UnitConversions.convert(10.0, 'deltaF', 'deltaC')) bypass_pipe = Model.add_pipe_adiabatic(model) out_pipe = Model.add_pipe_adiabatic(model) - loop.addSupplyBranchForComponent(bypass_pipe) - out_pipe.addToNode(loop.supplyOutletNode) + plant_loop.addSupplyBranchForComponent(bypass_pipe) + out_pipe.addToNode(plant_loop.supplyOutletNode) pump = Model.add_pump_variable_speed( model, name: "#{name} pump", rated_power: 0 ) - pump.addToNode(loop.supplyInletNode) + pump.addToNode(plant_loop.supplyInletNode) temp_schedule = Model.add_schedule_constant( model, @@ -2169,16 +2178,23 @@ def self.create_new_loop(model, t_set_c, eri_version, unit_multiplier) value: t_set_c ) setpoint_manager = OpenStudio::Model::SetpointManagerScheduled.new(model, temp_schedule) - setpoint_manager.addToNode(loop.supplyOutletNode) + setpoint_manager.addToNode(plant_loop.supplyOutletNode) - return loop + return plant_loop end - # TODO + # Gets the solar fraction, which is defined as the portion of total conventional hot water heating + # load (delivered energy plus tank standby losses) served by the solar thermal system. + # + # A value of zero will be returned unless all of these conditions are met: + # 1. There is a solar thermal system + # 2. The solar thermal system is attached to this water heater + # 3. The solar thermal system has simple inputs (i.e., solar fraction), as opposed to detailed + # inputs that would result in the solar thermal system being modeled explicitly in EnergyPlus. # # @param water_heating_system [HPXML::WaterHeatingSystem] The HPXML water heating system of interest # @param hpxml_bldg [HPXML::Building] HPXML Building object representing an individual dwelling unit - # @return [TODO] TODO + # @return [Double] Solar fraction or zero def self.get_water_heater_solar_fraction(water_heating_system, hpxml_bldg) return 0.0 if hpxml_bldg.solar_thermal_systems.size == 0 diff --git a/HPXMLtoOpenStudio/resources/xmlhelper.rb b/HPXMLtoOpenStudio/resources/xmlhelper.rb index 05b66f4b7a..b6a9a0b731 100644 --- a/HPXMLtoOpenStudio/resources/xmlhelper.rb +++ b/HPXMLtoOpenStudio/resources/xmlhelper.rb @@ -36,15 +36,18 @@ def self.insert_element(parent, element_name, index = 0, value = nil, datatype = parent.children.insert(index, added) end if not value.nil? - if datatype == :integer + case datatype + when :integer value = to_integer(value, parent, element_name) - elsif datatype == :float + when :float value = to_float(value, parent, element_name) - elsif datatype == :boolean + when :boolean value = to_boolean(value, parent, element_name) - elsif datatype != :string - # If value provided, datatype required - fail 'Unexpected datatype.' + else + if datatype != :string + # If value provided, datatype required + fail 'Unexpected datatype.' + end end added.inner_text = value.to_s end @@ -113,14 +116,17 @@ def self.get_value(parent, element_name, datatype) value = element.text - if datatype == :integer + case datatype + when :integer value = to_integer_or_nil(value, parent, element_name) - elsif datatype == :float + when :float value = to_float_or_nil(value, parent, element_name) - elsif datatype == :boolean + when :boolean value = to_boolean_or_nil(value, parent, element_name) - elsif datatype != :string - fail 'Unexpected datatype.' + else + if datatype != :string + fail 'Unexpected datatype.' + end end return value @@ -138,14 +144,17 @@ def self.get_values(parent, element_name, datatype) parent.xpath(element_name).each do |value| value = value.text - if datatype == :integer + case datatype + when :integer value = to_integer_or_nil(value, parent, element_name) - elsif datatype == :float + when :float value = to_float_or_nil(value, parent, element_name) - elsif datatype == :boolean + when :boolean value = to_boolean_or_nil(value, parent, element_name) - elsif datatype != :string - fail 'Unexpected datatype.' + else + if datatype != :string + fail 'Unexpected datatype.' + end end values << value diff --git a/HPXMLtoOpenStudio/tests/test_airflow.rb b/HPXMLtoOpenStudio/tests/test_airflow.rb index d67967fa04..da1c90dbb8 100644 --- a/HPXMLtoOpenStudio/tests/test_airflow.rb +++ b/HPXMLtoOpenStudio/tests/test_airflow.rb @@ -928,6 +928,26 @@ def test_duct_effective_r_values end end + def test_operational_0_occupants + args_hash = {} + args_hash['hpxml_path'] = File.absolute_path(File.join(@sample_files_path, 'base-residents-0.xml')) + model, _hpxml, _hpxml_bldg = _test_measure(args_hash) + + # Check no natural ventilation or whole house fan + program_values = get_ems_values(model.getEnergyManagementSystemPrograms, "#{Constants::ObjectTypeNaturalVentilation} program") + assert_equal(0, UnitConversions.convert(program_values['NVArea'].sum, 'cm^2', 'ft^2')) + assert_equal(0, UnitConversions.convert(program_values['WHF_Flow'].sum, 'm^3/s', 'cfm')) + + # Check no clothes dryer venting + program_values = get_ems_values(model.getEnergyManagementSystemPrograms, "#{Constants::ObjectTypeInfiltration} program") + assert_equal(0, UnitConversions.convert(program_values['Qdryer'].sum, 'm^3/s', 'cfm')) + + # Check no kitchen/bath local ventilation + program_values = get_ems_values(model.getEnergyManagementSystemPrograms, "#{Constants::ObjectTypeInfiltration} program") + assert_equal(0, UnitConversions.convert(program_values['Qrange'].sum, 'm^3/s', 'cfm')) + assert_equal(0, UnitConversions.convert(program_values['Qbath'].sum, 'm^3/s', 'cfm')) + end + def _test_measure(args_hash) # create an instance of the measure measure = HPXMLtoOpenStudio.new diff --git a/HPXMLtoOpenStudio/tests/test_hotwater_appliance.rb b/HPXMLtoOpenStudio/tests/test_hotwater_appliance.rb index 20baadd275..619be06570 100644 --- a/HPXMLtoOpenStudio/tests/test_hotwater_appliance.rb +++ b/HPXMLtoOpenStudio/tests/test_hotwater_appliance.rb @@ -952,6 +952,28 @@ def test_usage_multiplier assert_in_epsilon(1.0, get_oe_fractions(model, Constants::ObjectTypeGeneralWaterUseLatent)[1], 0.001) end + def test_operational_0_occupants + args_hash = {} + args_hash['hpxml_path'] = File.absolute_path(File.join(sample_files_dir, 'base-residents-0.xml')) + model, _hpxml, _hpxml_bldg = _test_measure(args_hash) + + # water use equipment hot water gal/day + assert_equal(0, get_wu_gpd(model, Constants::ObjectTypeClothesWasher)) + assert_equal(0, get_wu_gpd(model, Constants::ObjectTypeDishwasher)) + assert_equal(0, get_wu_gpd(model, Constants::ObjectTypeFixtures)) + assert_equal(0, get_wu_gpd(model, Constants::ObjectTypeDistributionWaste)) + + # electric equipment + assert_equal(0, get_ee_kwh_per_year(model, Constants::ObjectTypeClothesWasher)) + assert_equal(0, get_ee_kwh_per_year(model, Constants::ObjectTypeDishwasher)) + assert_equal(0, get_ee_kwh_per_year(model, Constants::ObjectTypeClothesDryer)) + assert_equal(0, get_ee_kwh_per_year(model, Constants::ObjectTypeCookingRange)) + + # other equipment + assert_equal(0, get_oe_kwh(model, Constants::ObjectTypeGeneralWaterUseSensible)) + assert_equal(0, get_oe_kwh(model, Constants::ObjectTypeGeneralWaterUseLatent)) + end + def test_operational_1_occupant args_hash = {} args_hash['hpxml_path'] = File.absolute_path(File.join(sample_files_dir, 'base-residents-1.xml')) diff --git a/HPXMLtoOpenStudio/tests/test_lighting.rb b/HPXMLtoOpenStudio/tests/test_lighting.rb index 9c92c57028..467f8acc41 100644 --- a/HPXMLtoOpenStudio/tests/test_lighting.rb +++ b/HPXMLtoOpenStudio/tests/test_lighting.rb @@ -133,6 +133,21 @@ def test_ceiling_fan assert_in_delta(200, get_kwh_per_year(model, Constants::ObjectTypeCeilingFan), 1.0) end + def test_operational_0_occupants + args_hash = {} + args_hash['hpxml_path'] = File.absolute_path(File.join(sample_files_dir, 'base-residents-0.xml')) + model, _hpxml, _hpxml_bldg = _test_measure(args_hash) + + # Check interior lighting + assert_equal(0.0, get_kwh_per_year(model, Constants::ObjectTypeLightingInterior)) + + # Check garage lighting + assert_equal(0.0, get_kwh_per_year(model, Constants::ObjectTypeLightingGarage)) + + # Check exterior lighting + assert_equal(0.0, get_kwh_per_year(model, Constants::ObjectTypeLightingExterior)) + end + def _test_measure(args_hash) # create an instance of the measure measure = HPXMLtoOpenStudio.new diff --git a/HPXMLtoOpenStudio/tests/test_miscloads.rb b/HPXMLtoOpenStudio/tests/test_miscloads.rb index b8e0121688..66e03f7376 100644 --- a/HPXMLtoOpenStudio/tests/test_miscloads.rb +++ b/HPXMLtoOpenStudio/tests/test_miscloads.rb @@ -195,7 +195,68 @@ def test_large_uncommon_loads2 assert_in_delta(55, therm_yr, 1.0) end - def test_operational_defaults + def test_operational_0_occupants + args_hash = {} + args_hash['hpxml_path'] = File.absolute_path(File.join(sample_files_dir, 'base-residents-0.xml')) + model, _hpxml, _hpxml_bldg = _test_measure(args_hash) + + # Check misc plug loads + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscPlugLoads) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check television + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscTelevision) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check vehicle + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscElectricVehicleCharging) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check well pump + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscWellPump) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check pool pump + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscPoolPump) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check pool heater + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscPoolHeater) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check permanent spa pump + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscPermanentSpaPump) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check permanent spa heater + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscPermanentSpaHeater) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check grill + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscGrill) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check lighting + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscLighting) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + + # Check fireplace + kwh_yr, therm_yr = get_kwh_therm_per_year(model, Constants::ObjectTypeMiscFireplace) + assert_equal(0, kwh_yr) + assert_equal(0, therm_yr) + end + + def test_operational_5_5_occupants args_hash = {} args_hash['hpxml_path'] = File.absolute_path(File.join(sample_files_dir, 'base-residents-5-5.xml')) model, _hpxml, _hpxml_bldg = _test_measure(args_hash) diff --git a/HPXMLtoOpenStudio/tests/test_validation.rb b/HPXMLtoOpenStudio/tests/test_validation.rb index 485f6105de..21f4bf08e0 100644 --- a/HPXMLtoOpenStudio/tests/test_validation.rb +++ b/HPXMLtoOpenStudio/tests/test_validation.rb @@ -270,79 +270,80 @@ def test_schema_schematron_error_messages all_expected_errors.each_with_index do |(error_case, expected_errors), i| puts "[#{i + 1}/#{all_expected_errors.size}] Testing #{error_case}..." # Create HPXML object - if ['boiler-invalid-afue'].include? error_case + case error_case + when 'boiler-invalid-afue' hpxml, hpxml_bldg = _create_hpxml('base-hvac-boiler-oil-only.xml') hpxml_bldg.heating_systems[0].heating_efficiency_afue *= 100.0 - elsif ['clothes-dryer-location'].include? error_case + when 'clothes-dryer-location' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.clothes_dryers[0].location = HPXML::LocationGarage - elsif ['clothes-washer-location'].include? error_case + when 'clothes-washer-location' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.clothes_washers[0].location = HPXML::LocationGarage - elsif ['cooking-range-location'].include? error_case + when 'cooking-range-location' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.cooking_ranges[0].location = HPXML::LocationGarage - elsif ['dehumidifier-fraction-served'].include? error_case + when 'dehumidifier-fraction-served' hpxml, hpxml_bldg = _create_hpxml('base-appliances-dehumidifier-multiple.xml') hpxml_bldg.dehumidifiers[-1].fraction_served = 0.6 - elsif ['dhw-frac-load-served'].include? error_case + when 'dhw-frac-load-served' hpxml, hpxml_bldg = _create_hpxml('base-dhw-multiple.xml') hpxml_bldg.water_heating_systems[0].fraction_dhw_load_served = 0.35 - elsif ['dhw-invalid-ef-tank'].include? error_case + when 'dhw-invalid-ef-tank' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.water_heating_systems[0].energy_factor = 1.0 - elsif ['dhw-invalid-uef-tank-heat-pump'].include? error_case + when 'dhw-invalid-uef-tank-heat-pump' hpxml, hpxml_bldg = _create_hpxml('base-dhw-tank-heat-pump-uef.xml') hpxml_bldg.water_heating_systems[0].uniform_energy_factor = 1.0 - elsif ['dishwasher-location'].include? error_case + when 'dishwasher-location' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.dishwashers[0].location = HPXML::LocationGarage - elsif ['duct-leakage-cfm25'].include? error_case + when 'duct-leakage-cfm25' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].duct_leakage_measurements[0].duct_leakage_value = -2 hpxml_bldg.hvac_distributions[0].duct_leakage_measurements[1].duct_leakage_value = -3 - elsif ['duct-leakage-cfm50'].include? error_case + when 'duct-leakage-cfm50' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ducts-leakage-cfm50.xml') hpxml_bldg.hvac_distributions[0].duct_leakage_measurements[0].duct_leakage_value = -2 hpxml_bldg.hvac_distributions[0].duct_leakage_measurements[1].duct_leakage_value = -3 - elsif ['duct-leakage-percent'].include? error_case + when 'duct-leakage-percent' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].duct_leakage_measurements[0].duct_leakage_units = HPXML::UnitsPercent hpxml_bldg.hvac_distributions[0].duct_leakage_measurements[1].duct_leakage_units = HPXML::UnitsPercent - elsif ['duct-location'].include? error_case + when 'duct-location' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].ducts[0].duct_location = HPXML::LocationGarage hpxml_bldg.hvac_distributions[0].ducts[1].duct_location = HPXML::LocationGarage - elsif ['duct-location-unconditioned-space'].include? error_case + when 'duct-location-unconditioned-space' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].ducts[0].duct_location = HPXML::LocationUnconditionedSpace hpxml_bldg.hvac_distributions[0].ducts[1].duct_location = HPXML::LocationUnconditionedSpace - elsif ['emissions-electricity-schedule'].include? error_case + when 'emissions-electricity-schedule' hpxml, hpxml_bldg = _create_hpxml('base-misc-emissions.xml') hpxml.header.emissions_scenarios[0].elec_schedule_number_of_header_rows = -1 hpxml.header.emissions_scenarios[0].elec_schedule_column_number = 0 - elsif ['enclosure-attic-missing-roof'].include? error_case + when 'enclosure-attic-missing-roof' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.roofs.reverse_each do |roof| roof.delete end - elsif ['enclosure-basement-missing-exterior-foundation-wall'].include? error_case + when 'enclosure-basement-missing-exterior-foundation-wall' hpxml, hpxml_bldg = _create_hpxml('base-foundation-unconditioned-basement.xml') hpxml_bldg.foundation_walls.reverse_each do |foundation_wall| foundation_wall.delete end - elsif ['enclosure-basement-missing-slab'].include? error_case + when 'enclosure-basement-missing-slab' hpxml, hpxml_bldg = _create_hpxml('base-foundation-unconditioned-basement.xml') hpxml_bldg.slabs.reverse_each do |slab| slab.delete end - elsif ['enclosure-floor-area-exceeds-cfa'].include? error_case + when 'enclosure-floor-area-exceeds-cfa' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.building_construction.conditioned_floor_area = 1348.8 - elsif ['enclosure-floor-area-exceeds-cfa2'].include? error_case + when 'enclosure-floor-area-exceeds-cfa2' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit.xml') hpxml_bldg.building_construction.conditioned_floor_area = 898.8 - elsif ['enclosure-garage-missing-exterior-wall'].include? error_case + when 'enclosure-garage-missing-exterior-wall' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-garage.xml') hpxml_bldg.walls.select { |w| w.interior_adjacent_to == HPXML::LocationGarage && @@ -350,7 +351,7 @@ def test_schema_schematron_error_messages }.reverse_each do |wall| wall.delete end - elsif ['enclosure-garage-missing-roof-ceiling'].include? error_case + when 'enclosure-garage-missing-roof-ceiling' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-garage.xml') hpxml_bldg.floors.select { |w| w.interior_adjacent_to == HPXML::LocationGarage && @@ -358,75 +359,75 @@ def test_schema_schematron_error_messages }.reverse_each do |floor| floor.delete end - elsif ['enclosure-garage-missing-slab'].include? error_case + when 'enclosure-garage-missing-slab' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-garage.xml') hpxml_bldg.slabs.select { |w| w.interior_adjacent_to == HPXML::LocationGarage }.reverse_each do |slab| slab.delete end - elsif ['enclosure-conditioned-missing-ceiling-roof'].include? error_case + when 'enclosure-conditioned-missing-ceiling-roof' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.floors.reverse_each do |floor| floor.delete end - elsif ['enclosure-conditioned-missing-exterior-wall'].include? error_case + when 'enclosure-conditioned-missing-exterior-wall' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.walls.reverse_each do |wall| next unless wall.interior_adjacent_to == HPXML::LocationConditionedSpace wall.delete end - elsif ['enclosure-conditioned-missing-floor-slab'].include? error_case + when 'enclosure-conditioned-missing-floor-slab' hpxml, hpxml_bldg = _create_hpxml('base-foundation-slab.xml') hpxml_bldg.slabs[0].delete - elsif ['frac-sensible-latent-fuel-load-values'].include? error_case + when 'frac-sensible-latent-fuel-load-values' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml_bldg.fuel_loads[0].frac_sensible = -0.1 hpxml_bldg.fuel_loads[0].frac_latent = -0.1 - elsif ['frac-sensible-latent-fuel-load-presence'].include? error_case + when 'frac-sensible-latent-fuel-load-presence' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml_bldg.fuel_loads[0].frac_sensible = 1.0 hpxml_bldg.fuel_loads[0].frac_latent = nil - elsif ['frac-sensible-latent-plug-load-values'].include? error_case + when 'frac-sensible-latent-plug-load-values' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml_bldg.plug_loads[0].frac_sensible = -0.1 hpxml_bldg.plug_loads[0].frac_latent = -0.1 - elsif ['frac-sensible-latent-plug-load-presence'].include? error_case + when 'frac-sensible-latent-plug-load-presence' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml_bldg.plug_loads[0].frac_latent = 1.0 hpxml_bldg.plug_loads[0].frac_sensible = nil - elsif ['frac-total-fuel-load'].include? error_case + when 'frac-total-fuel-load' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml_bldg.fuel_loads[0].frac_sensible = 0.8 hpxml_bldg.fuel_loads[0].frac_latent = 0.3 - elsif ['frac-total-plug-load'].include? error_case + when 'frac-total-plug-load' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml_bldg.plug_loads[1].frac_latent = 0.245 - elsif ['furnace-invalid-afue'].include? error_case + when 'furnace-invalid-afue' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.heating_systems[0].heating_efficiency_afue *= 100.0 - elsif ['generator-number-of-bedrooms-served'].include? error_case + when 'generator-number-of-bedrooms-served' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-generator.xml') hpxml_bldg.generators[0].number_of_bedrooms_served = 3 - elsif ['generator-output-greater-than-consumption'].include? error_case + when 'generator-output-greater-than-consumption' hpxml, hpxml_bldg = _create_hpxml('base-misc-generators.xml') hpxml_bldg.generators[0].annual_consumption_kbtu = 1500 - elsif ['heat-pump-backup-sizing'].include? error_case + when 'heat-pump-backup-sizing' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.header.heat_pump_backup_sizing_methodology = 'foobar' - elsif ['heat-pump-separate-backup-inputs'].include? error_case + when 'heat-pump-separate-backup-inputs' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-var-speed-backup-furnace.xml') hpxml_bldg.heat_pumps[0].backup_heating_capacity = 12345 hpxml_bldg.heat_pumps[0].backup_heating_efficiency_afue = 0.8 hpxml_bldg.heat_pumps[0].backup_heating_autosizing_factor = 1.2 - elsif ['heat-pump-capacity-17f'].include? error_case + when 'heat-pump-capacity-17f' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.heat_pumps[0].heating_capacity_17F = hpxml_bldg.heat_pumps[0].heating_capacity + 1000.0 hpxml_bldg.heat_pumps[0].heating_capacity_retention_fraction = nil hpxml_bldg.heat_pumps[0].heating_capacity_retention_temp = nil - elsif ['heat-pump-lockout-temperatures'].include? error_case + when 'heat-pump-lockout-temperatures' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-lockout-temperatures.xml') hpxml_bldg.heat_pumps[0].compressor_lockout_temp = hpxml_bldg.heat_pumps[0].backup_heating_lockout_temp + 1 - elsif ['heat-pump-multiple-backup-systems'].include? error_case + when 'heat-pump-multiple-backup-systems' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-var-speed-backup-boiler.xml') hpxml_bldg.heating_systems << hpxml_bldg.heating_systems[0].dup hpxml_bldg.heating_systems[-1].id = 'HeatingSystem2' @@ -436,13 +437,13 @@ def test_schema_schematron_error_messages hpxml_bldg.heat_pumps[-1].id = 'HeatPump2' hpxml_bldg.heat_pumps[-1].primary_heating_system = false hpxml_bldg.heat_pumps[-1].primary_cooling_system = false - elsif ['hvac-detailed-performance-not-variable-speed'].include? error_case + when 'hvac-detailed-performance-not-variable-speed' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-var-speed-detailed-performance.xml') hpxml_bldg.heat_pumps[0].compressor_type = HPXML::HVACCompressorTypeTwoStage - elsif ['hvac-distribution-return-duct-leakage-missing'].include? error_case + when 'hvac-distribution-return-duct-leakage-missing' hpxml, hpxml_bldg = _create_hpxml('base-hvac-evap-cooler-only-ducted.xml') hpxml_bldg.hvac_distributions[0].duct_leakage_measurements[-1].delete - elsif ['hvac-frac-load-served'].include? error_case + when 'hvac-frac-load-served' hpxml, hpxml_bldg = _create_hpxml('base-hvac-multiple.xml') hpxml_bldg.heating_systems[0].fraction_heat_load_served += 0.1 hpxml_bldg.cooling_systems[0].fraction_cool_load_served += 0.2 @@ -450,22 +451,22 @@ def test_schema_schematron_error_messages hpxml_bldg.cooling_systems[0].primary_system = true hpxml_bldg.heat_pumps[-1].primary_heating_system = false hpxml_bldg.heat_pumps[-1].primary_cooling_system = false - elsif ['hvac-research-features-timestep-ten-mins'].include? error_case + when 'hvac-research-features-timestep-ten-mins' hpxml, _hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml.header.timestep = 10 - elsif ['hvac-research-features-timestep-missing'].include? error_case + when 'hvac-research-features-timestep-missing' hpxml, _hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml.header.timestep = nil - elsif ['hvac-research-features-onoff-thermostat-heat-load-fraction-partial'].include? error_case + when 'hvac-research-features-onoff-thermostat-heat-load-fraction-partial' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml_bldg.heat_pumps[0].fraction_heat_load_served = 0.5 - elsif ['hvac-research-features-onoff-thermostat-cool-load-fraction-partial'].include? error_case + when 'hvac-research-features-onoff-thermostat-cool-load-fraction-partial' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml_bldg.heat_pumps[0].fraction_cool_load_served = 0.5 - elsif ['hvac-research-features-onoff-thermostat-negative-value'].include? error_case + when 'hvac-research-features-onoff-thermostat-negative-value' hpxml, _hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml.header.hvac_onoff_thermostat_deadband = -1.0 - elsif ['hvac-research-features-onoff-thermostat-two-heat-pumps'].include? error_case + when 'hvac-research-features-onoff-thermostat-two-heat-pumps' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml_bldg.heat_pumps[0].fraction_cool_load_served = 0.5 hpxml_bldg.heat_pumps[0].fraction_heat_load_served = 0.5 @@ -473,80 +474,80 @@ def test_schema_schematron_error_messages hpxml_bldg.heat_pumps[-1].id = 'HeatPump2' hpxml_bldg.heat_pumps[-1].primary_heating_system = false hpxml_bldg.heat_pumps[-1].primary_cooling_system = false - elsif ['hvac-gshp-invalid-bore-config'].include? error_case + when 'hvac-gshp-invalid-bore-config' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump-detailed-geothermal-loop.xml') hpxml_bldg.geothermal_loops[0].bore_config = 'Invalid' - elsif ['hvac-gshp-invalid-bore-depth-low'].include? error_case + when 'hvac-gshp-invalid-bore-depth-low' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump-detailed-geothermal-loop.xml') hpxml_bldg.geothermal_loops[0].bore_length = 78 - elsif ['hvac-gshp-invalid-bore-depth-high'].include? error_case + when 'hvac-gshp-invalid-bore-depth-high' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump-detailed-geothermal-loop.xml') hpxml_bldg.geothermal_loops[0].bore_length = 501 - elsif ['hvac-gshp-autosized-count-not-rectangle'].include? error_case + when 'hvac-gshp-autosized-count-not-rectangle' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump-detailed-geothermal-loop.xml') hpxml_bldg.geothermal_loops[0].num_bore_holes = nil - elsif ['hvac-location-heating-system'].include? error_case + when 'hvac-location-heating-system' hpxml, hpxml_bldg = _create_hpxml('base-hvac-boiler-oil-only.xml') hpxml_bldg.heating_systems[0].location = HPXML::LocationBasementUnconditioned - elsif ['hvac-location-cooling-system'].include? error_case + when 'hvac-location-cooling-system' hpxml, hpxml_bldg = _create_hpxml('base-hvac-central-ac-only-1-speed.xml') hpxml_bldg.cooling_systems[0].location = HPXML::LocationBasementUnconditioned - elsif ['hvac-location-heat-pump'].include? error_case + when 'hvac-location-heat-pump' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.heat_pumps[0].location = HPXML::LocationBasementUnconditioned - elsif ['hvac-msac-not-var-speed'].include? error_case + when 'hvac-msac-not-var-speed' hpxml, hpxml_bldg = _create_hpxml('base-hvac-mini-split-air-conditioner-only-ductless.xml') hpxml_bldg.cooling_systems[0].compressor_type = HPXML::HVACCompressorTypeTwoStage - elsif ['hvac-mshp-not-var-speed'].include? error_case + when 'hvac-mshp-not-var-speed' hpxml, hpxml_bldg = _create_hpxml('base-hvac-mini-split-heat-pump-ductless.xml') hpxml_bldg.heat_pumps[0].compressor_type = HPXML::HVACCompressorTypeSingleStage - elsif ['hvac-shr-low'].include? error_case + when 'hvac-shr-low' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.cooling_systems[0].cooling_shr = 0.4 - elsif ['hvac-sizing-humidity-setpoint'].include? error_case + when 'hvac-sizing-humidity-setpoint' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.header.manualj_humidity_setpoint = 50 - elsif ['hvac-sizing-daily-temp-range'].include? error_case + when 'hvac-sizing-daily-temp-range' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.header.manualj_daily_temp_range = 'foobar' - elsif ['hvac-negative-crankcase-heater-watts'].include? error_case + when 'hvac-negative-crankcase-heater-watts' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.cooling_systems[0].crankcase_heater_watts = -10 - elsif ['incomplete-integrated-heating'].include? error_case + when 'incomplete-integrated-heating' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ptac-with-heating-electricity.xml') hpxml_bldg.cooling_systems[0].integrated_heating_system_fraction_heat_load_served = nil - elsif ['invalid-airflow-defect-ratio'].include? error_case + when 'invalid-airflow-defect-ratio' hpxml, hpxml_bldg = _create_hpxml('base-hvac-mini-split-heat-pump-ductless.xml') hpxml_bldg.heat_pumps[0].airflow_defect_ratio = -0.25 - elsif ['invalid-assembly-effective-rvalue'].include? error_case + when 'invalid-assembly-effective-rvalue' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.walls[0].insulation_assembly_r_value = 0.0 - elsif ['invalid-battery-capacities-ah'].include? error_case + when 'invalid-battery-capacities-ah' hpxml, hpxml_bldg = _create_hpxml('base-pv-battery-ah.xml') hpxml_bldg.batteries[0].usable_capacity_ah = hpxml_bldg.batteries[0].nominal_capacity_ah - elsif ['invalid-battery-capacities-kwh'].include? error_case + when 'invalid-battery-capacities-kwh' hpxml, hpxml_bldg = _create_hpxml('base-pv-battery.xml') hpxml_bldg.batteries[0].usable_capacity_kwh = hpxml_bldg.batteries[0].nominal_capacity_kwh - elsif ['invalid-calendar-year-low'].include? error_case + when 'invalid-calendar-year-low' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml.header.sim_calendar_year = 1575 - elsif ['invalid-calendar-year-high'].include? error_case + when 'invalid-calendar-year-high' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml.header.sim_calendar_year = 20000 - elsif ['invalid-clothes-dryer-cef'].include? error_case + when 'invalid-clothes-dryer-cef' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.clothes_dryers[0].combined_energy_factor = 0 - elsif ['invalid-clothes-washer-imef'].include? error_case + when 'invalid-clothes-washer-imef' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.clothes_washers[0].integrated_modified_energy_factor = 0 - elsif ['invalid-cfis-addtl-runtime-mode'].include? error_case + when 'invalid-cfis-addtl-runtime-mode' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-cfis-control-type-timer.xml') hpxml_bldg.ventilation_fans[0].cfis_addtl_runtime_operating_mode = HPXML::CFISModeNone hpxml_bldg.ventilation_fans[0].fan_power = nil - elsif ['invalid-dishwasher-ler'].include? error_case + when 'invalid-dishwasher-ler' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.dishwashers[0].label_electric_rate = 0 - elsif ['invalid-duct-area-fractions'].include? error_case + when 'invalid-duct-area-fractions' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ducts-area-fractions.xml') hpxml_bldg.hvac_distributions[0].ducts[0].duct_surface_area = nil hpxml_bldg.hvac_distributions[0].ducts[1].duct_surface_area = nil @@ -556,41 +557,41 @@ def test_schema_schematron_error_messages hpxml_bldg.hvac_distributions[0].ducts[1].duct_fraction_area = 0.65 hpxml_bldg.hvac_distributions[0].ducts[2].duct_fraction_area = 0.15 hpxml_bldg.hvac_distributions[0].ducts[3].duct_fraction_area = 0.15 - elsif ['invalid-facility-type'].include? error_case + when 'invalid-facility-type' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-laundry-room.xml') hpxml_bldg.building_construction.residential_facility_type = HPXML::ResidentialTypeSFD - elsif ['invalid-foundation-wall-properties'].include? error_case + when 'invalid-foundation-wall-properties' hpxml, hpxml_bldg = _create_hpxml('base-foundation-unconditioned-basement-wall-insulation.xml') hpxml_bldg.foundation_walls[0].depth_below_grade = 9.0 hpxml_bldg.foundation_walls[0].insulation_interior_distance_to_top = 12.0 hpxml_bldg.foundation_walls[0].insulation_interior_distance_to_bottom = 10.0 - elsif ['invalid-ground-conductivity'].include? error_case + when 'invalid-ground-conductivity' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.site.ground_conductivity = 0.0 - elsif ['invalid-ground-diffusivity'].include? error_case + when 'invalid-ground-diffusivity' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.site.ground_diffusivity = 0.0 - elsif ['invalid-heat-pump-capacity-retention'].include? error_case + when 'invalid-heat-pump-capacity-retention' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.heat_pumps[0].heating_capacity_17F = nil hpxml_bldg.heat_pumps[0].heating_capacity_retention_fraction = 1.5 hpxml_bldg.heat_pumps[0].heating_capacity_retention_temp = 30 - elsif ['invalid-heat-pump-capacity-retention2'].include? error_case + when 'invalid-heat-pump-capacity-retention2' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.heat_pumps[0].heating_capacity_17F = nil hpxml_bldg.heat_pumps[0].heating_capacity_retention_fraction = -1 hpxml_bldg.heat_pumps[0].heating_capacity_retention_temp = 5 - elsif ['invalid-hvac-installation-quality'].include? error_case + when 'invalid-hvac-installation-quality' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.heat_pumps[0].airflow_defect_ratio = -99 hpxml_bldg.heat_pumps[0].charge_defect_ratio = -99 - elsif ['invalid-hvac-installation-quality2'].include? error_case + when 'invalid-hvac-installation-quality2' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.heat_pumps[0].airflow_defect_ratio = 99 hpxml_bldg.heat_pumps[0].charge_defect_ratio = 99 - elsif ['invalid-id2'].include? error_case + when 'invalid-id2' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-skylights.xml') - elsif ['invalid-input-parameters'].include? error_case + when 'invalid-input-parameters' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml.header.transaction = 'modify' hpxml_bldg.site.site_type = 'mountain' @@ -601,15 +602,15 @@ def test_schema_schematron_error_messages hpxml_bldg.roofs[0].azimuth = 365 hpxml_bldg.dishwashers[0].rated_annual_kwh = nil hpxml_bldg.dishwashers[0].energy_factor = 5.1 - elsif ['invalid-insulation-top'].include? error_case + when 'invalid-insulation-top' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.foundation_walls[0].insulation_interior_distance_to_top = -0.5 - elsif ['invalid-integrated-heating'].include? error_case + when 'invalid-integrated-heating' hpxml, hpxml_bldg = _create_hpxml('base-hvac-central-ac-only-1-speed.xml') hpxml_bldg.cooling_systems[0].integrated_heating_system_fuel = HPXML::FuelTypeElectricity hpxml_bldg.cooling_systems[0].integrated_heating_system_efficiency_percent = 0.98 hpxml_bldg.cooling_systems[0].integrated_heating_system_fraction_heat_load_served = 1.0 - elsif ['invalid-lighting-groups'].include? error_case + when 'invalid-lighting-groups' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-garage.xml') [HPXML::LocationInterior, HPXML::LocationExterior, HPXML::LocationGarage].each do |ltg_loc| hpxml_bldg.lighting_groups.each do |lg| @@ -619,7 +620,7 @@ def test_schema_schematron_error_messages break end end - elsif ['invalid-lighting-groups2'].include? error_case + when 'invalid-lighting-groups2' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-garage.xml') [HPXML::LocationInterior, HPXML::LocationExterior, HPXML::LocationGarage].each do |ltg_loc| hpxml_bldg.lighting_groups.each do |lg| @@ -631,75 +632,75 @@ def test_schema_schematron_error_messages break end end - elsif ['invalid-natvent-availability'].include? error_case + when 'invalid-natvent-availability' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.header.natvent_days_per_week = 8 - elsif ['invalid-natvent-availability2'].include? error_case + when 'invalid-natvent-availability2' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.header.natvent_days_per_week = -1 - elsif ['invalid-number-of-bedrooms-served-pv'].include? error_case + when 'invalid-number-of-bedrooms-served-pv' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-pv.xml') hpxml_bldg.pv_systems[0].number_of_bedrooms_served = 3 - elsif ['invalid-number-of-bedrooms-served-recirc'].include? error_case + when 'invalid-number-of-bedrooms-served-recirc' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-water-heater-recirc.xml') hpxml_bldg.hot_water_distributions[0].shared_recirculation_number_of_bedrooms_served = 3 - elsif ['invalid-number-of-bedrooms-served-water-heater'].include? error_case + when 'invalid-number-of-bedrooms-served-water-heater' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-water-heater.xml') hpxml_bldg.water_heating_systems[0].number_of_bedrooms_served = 3 - elsif ['invalid-number-of-conditioned-floors'].include? error_case + when 'invalid-number-of-conditioned-floors' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.building_construction.number_of_conditioned_floors_above_grade = 3 - elsif ['invalid-number-of-conditioned-floors-above-grade'].include? error_case + when 'invalid-number-of-conditioned-floors-above-grade' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.building_construction.number_of_conditioned_floors_above_grade = 0 - elsif ['invalid-pilot-light-heating-system'].include? error_case + when 'invalid-pilot-light-heating-system' hpxml, hpxml_bldg = _create_hpxml('base-hvac-floor-furnace-propane-only.xml') hpxml_bldg.heating_systems[0].heating_system_fuel = HPXML::FuelTypeElectricity - elsif ['invalid-soil-type'].include? error_case + when 'invalid-soil-type' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.site.soil_type = HPXML::SiteSoilTypeOther - elsif ['invalid-shared-vent-in-unit-flowrate'].include? error_case + when 'invalid-shared-vent-in-unit-flowrate' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-mechvent.xml') hpxml_bldg.ventilation_fans[0].rated_flow_rate = 80 - elsif ['invalid-timestep'].include? error_case + when 'invalid-timestep' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml.header.timestep = 45 - elsif ['invalid-timezone-utcoffset-low'].include? error_case + when 'invalid-timezone-utcoffset-low' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.time_zone_utc_offset = -13 - elsif ['invalid-timezone-utcoffset-high'].include? error_case + when 'invalid-timezone-utcoffset-high' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.time_zone_utc_offset = 15 - elsif ['invalid-ventilation-fan'].include? error_case + when 'invalid-ventilation-fan' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-exhaust.xml') hpxml_bldg.ventilation_fans[0].used_for_garage_ventilation = true - elsif ['invalid-ventilation-recovery'].include? error_case + when 'invalid-ventilation-recovery' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-exhaust.xml') hpxml_bldg.ventilation_fans[0].sensible_recovery_efficiency = 0.72 hpxml_bldg.ventilation_fans[0].total_recovery_efficiency = 0.48 - elsif ['invalid-water-heater-heating-capacity'].include? error_case + when 'invalid-water-heater-heating-capacity' hpxml, hpxml_bldg = _create_hpxml('base-dhw-tank-gas.xml') hpxml_bldg.water_heating_systems[0].heating_capacity = 0 - elsif ['invalid-water-heater-heating-capacity2'].include? error_case + when 'invalid-water-heater-heating-capacity2' hpxml, hpxml_bldg = _create_hpxml('base-dhw-tank-heat-pump.xml') hpxml_bldg.water_heating_systems[0].heating_capacity = 0 - elsif ['invalid-window-height'].include? error_case + when 'invalid-window-height' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-overhangs.xml') hpxml_bldg.windows[1].overhangs_distance_to_bottom_of_window = 1.0 - elsif ['leakiness-description-missing-year-built'].include? error_case + when 'leakiness-description-missing-year-built' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-infil-leakiness-description.xml') hpxml_bldg.building_construction.year_built = nil - elsif ['lighting-fractions'].include? error_case + when 'lighting-fractions' hpxml, hpxml_bldg = _create_hpxml('base.xml') int_cfl = hpxml_bldg.lighting_groups.find { |lg| lg.location == HPXML::LocationInterior && lg.lighting_type == HPXML::LightingTypeCFL } int_cfl.fraction_of_units_in_location = 0.8 - elsif ['manufactured-home-reference-duct'].include? error_case + when 'manufactured-home-reference-duct' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].ducts[1].duct_location = HPXML::LocationManufacturedHomeBelly - elsif ['manufactured-home-reference-water-heater'].include? error_case + when 'manufactured-home-reference-water-heater' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.water_heating_systems[0].location = HPXML::LocationManufacturedHomeBelly - elsif ['manufactured-home-reference-floor'].include? error_case + when 'manufactured-home-reference-floor' hpxml, hpxml_bldg = _create_hpxml('base-foundation-vented-crawlspace.xml') hpxml_bldg.floors.each do |floor| if floor.exterior_adjacent_to == HPXML::LocationCrawlspaceVented @@ -707,97 +708,97 @@ def test_schema_schematron_error_messages break end end - elsif ['missing-attached-to-space-wall'].include? error_case + when 'missing-attached-to-space-wall' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.walls.find { |s| s.interior_adjacent_to == HPXML::LocationConditionedSpace }.attached_to_space_idref = nil - elsif ['missing-attached-to-space-slab'].include? error_case + when 'missing-attached-to-space-slab' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.slabs.find { |s| s.interior_adjacent_to == HPXML::LocationBasementConditioned }.attached_to_space_idref = nil - elsif ['missing-attached-to-zone'].include? error_case + when 'missing-attached-to-zone' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.hvac_systems[0].attached_to_zone_idref = nil - elsif ['missing-capacity-detailed-performance'].include? error_case + when 'missing-capacity-detailed-performance' hpxml, hpxml_bldg = _create_hpxml('base-hvac-mini-split-heat-pump-ductless-detailed-performance.xml') hpxml_bldg.heat_pumps[0].cooling_capacity = nil hpxml_bldg.heat_pumps[0].heating_capacity = nil - elsif ['missing-cfis-supplemental-fan'].include? error_case + when 'missing-cfis-supplemental-fan' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-cfis-supplemental-fan-exhaust.xml') hpxml_bldg.ventilation_fans[1].delete - elsif ['missing-distribution-cfa-served'].include? error_case + when 'missing-distribution-cfa-served' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].ducts[1].duct_surface_area = nil hpxml_bldg.hvac_distributions[0].ducts[1].duct_location = nil - elsif ['missing-duct-area'].include? error_case + when 'missing-duct-area' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].conditioned_floor_area_served = hpxml_bldg.building_construction.conditioned_floor_area hpxml_bldg.hvac_distributions[0].ducts[1].duct_surface_area = nil - elsif ['missing-duct-location'].include? error_case + when 'missing-duct-location' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].ducts[1].duct_location = nil - elsif ['missing-elements'].include? error_case + when 'missing-elements' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.building_construction.number_of_conditioned_floors = nil hpxml_bldg.building_construction.conditioned_floor_area = nil - elsif ['missing-epw-filepath-and-zipcode'].include? error_case + when 'missing-epw-filepath-and-zipcode' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.climate_and_risk_zones.weather_station_epw_filepath = nil - elsif ['missing-skylight-floor'].include? error_case + when 'missing-skylight-floor' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-skylights.xml') hpxml_bldg.skylights[0].attached_to_floor_idref = nil - elsif ['multifamily-reference-appliance'].include? error_case + when 'multifamily-reference-appliance' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.clothes_washers[0].location = HPXML::LocationOtherHousingUnit - elsif ['multifamily-reference-duct'].include? error_case + when 'multifamily-reference-duct' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].ducts[0].duct_location = HPXML::LocationOtherMultifamilyBufferSpace - elsif ['multifamily-reference-surface'].include? error_case + when 'multifamily-reference-surface' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.floors << hpxml_bldg.floors[0].dup hpxml_bldg.floors[1].id = "Floor#{hpxml_bldg.floors.size}" hpxml_bldg.floors[1].insulation_id = "FloorInsulation#{hpxml_bldg.floors.size}" hpxml_bldg.floors[1].exterior_adjacent_to = HPXML::LocationOtherHeatedSpace hpxml_bldg.floors[1].floor_or_ceiling = HPXML::FloorOrCeilingCeiling - elsif ['multifamily-reference-water-heater'].include? error_case + when 'multifamily-reference-water-heater' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.water_heating_systems[0].location = HPXML::LocationOtherNonFreezingSpace - elsif ['negative-autosizing-factors'].include? error_case + when 'negative-autosizing-factors' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-autosize-factor.xml') hpxml_bldg.heat_pumps[0].heating_autosizing_factor = -0.5 hpxml_bldg.heat_pumps[0].cooling_autosizing_factor = -1.2 hpxml_bldg.heat_pumps[0].backup_heating_autosizing_factor = -0.1 - elsif ['refrigerator-location'].include? error_case + when 'refrigerator-location' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.refrigerators[0].location = HPXML::LocationGarage - elsif ['refrigerator-schedule'].include? error_case + when 'refrigerator-schedule' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.refrigerators[0].weekday_fractions = '0.040, 0.039, 0.038, 0.037, 0.036, 0.036, 0.038, 0.040, 0.041, 0.041, 0.040, 0.040, 0.042, 0.042, 0.042, 0.041, 0.044, 0.048, 0.050, 0.048, 0.047, 0.046, 0.044, 0.041' hpxml_bldg.refrigerators[0].constant_coefficients = '-0.487, -0.340, -0.370, -0.361, -0.515, -0.684, -0.471, -0.159, -0.079, -0.417, -0.411, -0.386, -0.240, -0.314, -0.160, -0.121, -0.469, -0.412, -0.091, 0.077, -0.118, -0.247, -0.445, -0.544' - elsif ['solar-fraction-one'].include? error_case + when 'solar-fraction-one' hpxml, hpxml_bldg = _create_hpxml('base-dhw-solar-fraction.xml') hpxml_bldg.solar_thermal_systems[0].solar_fraction = 1.0 - elsif ['sum-space-floor-area'].include? error_case + when 'sum-space-floor-area' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.conditioned_spaces.each do |space| space.floor_area /= 2.0 end - elsif ['sum-space-floor-area2'].include? error_case + when 'sum-space-floor-area2' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.conditioned_spaces.each do |space| space.floor_area *= 2.0 end - elsif ['water-heater-location'].include? error_case + when 'water-heater-location' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.water_heating_systems[0].location = HPXML::LocationCrawlspaceVented - elsif ['water-heater-location-other'].include? error_case + when 'water-heater-location-other' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.water_heating_systems[0].location = HPXML::LocationUnconditionedSpace - elsif ['water-heater-recovery-efficiency'].include? error_case + when 'water-heater-recovery-efficiency' hpxml, hpxml_bldg = _create_hpxml('base-dhw-tank-gas.xml') hpxml_bldg.water_heating_systems[0].recovery_efficiency = hpxml_bldg.water_heating_systems[0].energy_factor - elsif ['wrong-infiltration-method-blower-door'].include? error_case + when 'wrong-infiltration-method-blower-door' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-infil-leakiness-description.xml') hpxml_bldg.header.manualj_infiltration_method = HPXML::ManualJInfiltrationMethodBlowerDoor - elsif ['wrong-infiltration-method-default-table'].include? error_case + when 'wrong-infiltration-method-default-table' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.header.manualj_infiltration_method = HPXML::ManualJInfiltrationMethodDefaultTable else @@ -807,7 +808,7 @@ def test_schema_schematron_error_messages hpxml_doc = hpxml.to_doc() # Perform additional raw XML manipulation - if ['invalid-id2'].include? error_case + if error_case == 'invalid-id2' element = XMLHelper.get_element(hpxml_doc, '/HPXML/Building/BuildingDetails/Enclosure/Skylights/Skylight/SystemIdentifier') XMLHelper.delete_attribute(element, 'id') end @@ -907,19 +908,20 @@ def test_schema_schematron_warning_messages all_expected_warnings.each_with_index do |(warning_case, expected_warnings), i| puts "[#{i + 1}/#{all_expected_warnings.size}] Testing #{warning_case}..." # Create HPXML object - if ['battery-pv-output-power-low'].include? warning_case + case warning_case + when 'battery-pv-output-power-low' hpxml, hpxml_bldg = _create_hpxml('base-pv-battery.xml') hpxml_bldg.batteries[0].rated_power_output = 0.1 hpxml_bldg.pv_systems[0].max_power_output = 0.1 hpxml_bldg.pv_systems[1].max_power_output = 0.1 - elsif ['dhw-capacities-low'].include? warning_case + when 'dhw-capacities-low' hpxml, hpxml_bldg = _create_hpxml('base-dhw-multiple.xml') hpxml_bldg.water_heating_systems.each do |water_heating_system| if [HPXML::WaterHeaterTypeStorage].include? water_heating_system.water_heater_type water_heating_system.heating_capacity = 0.1 end end - elsif ['dhw-efficiencies-low'].include? warning_case + when 'dhw-efficiencies-low' hpxml, hpxml_bldg = _create_hpxml('base-dhw-multiple.xml') hpxml_bldg.water_heating_systems.each do |water_heating_system| if [HPXML::WaterHeaterTypeStorage, @@ -927,33 +929,33 @@ def test_schema_schematron_warning_messages water_heating_system.energy_factor = 0.1 end end - elsif ['dhw-setpoint-low'].include? warning_case + when 'dhw-setpoint-low' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.water_heating_systems[0].temperature = 100 - elsif ['erv-atre-low'].include? warning_case + when 'erv-atre-low' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-erv-atre-asre.xml') hpxml_bldg.ventilation_fans[0].total_recovery_efficiency_adjusted = 0.1 - elsif ['fuel-load-type-other'].include? warning_case + when 'fuel-load-type-other' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml_bldg.fuel_loads[0].fuel_load_type = HPXML::FuelLoadTypeOther - elsif ['erv-tre-low'].include? warning_case + when 'erv-tre-low' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-erv.xml') hpxml_bldg.ventilation_fans[0].total_recovery_efficiency = 0.1 - elsif ['garage-ventilation'].include? warning_case + when 'garage-ventilation' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.ventilation_fans.add(id: 'VentilationFan1', used_for_garage_ventilation: true) - elsif ['heat-pump-low-backup-switchover-temp'].include? warning_case + when 'heat-pump-low-backup-switchover-temp' hpxml, hpxml_bldg = _create_hpxml('base-hvac-dual-fuel-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.heat_pumps[0].backup_heating_switchover_temp = 25.0 - elsif ['heat-pump-low-backup-lockout-temp'].include? warning_case + when 'heat-pump-low-backup-lockout-temp' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-lockout-temperatures.xml') hpxml_bldg.heat_pumps[0].backup_heating_lockout_temp = 25.0 - elsif ['hvac-dse-low'].include? warning_case + when 'hvac-dse-low' hpxml, hpxml_bldg = _create_hpxml('base-hvac-dse.xml') hpxml_bldg.hvac_distributions[0].annual_heating_dse = 0.1 hpxml_bldg.hvac_distributions[0].annual_cooling_dse = 0.1 - elsif ['hvac-capacities-low'].include? warning_case + when 'hvac-capacities-low' hpxml, hpxml_bldg = _create_hpxml('base-hvac-multiple.xml') hpxml_bldg.hvac_systems.each do |hvac_system| if hvac_system.is_a? HPXML::HeatingSystem @@ -966,97 +968,100 @@ def test_schema_schematron_warning_messages hvac_system.backup_heating_capacity = 0.1 end end - elsif ['hvac-efficiencies-low'].include? warning_case + when 'hvac-efficiencies-low' hpxml, hpxml_bldg = _create_hpxml('base-hvac-multiple.xml') hpxml_bldg.hvac_systems.each do |hvac_system| if hvac_system.is_a? HPXML::HeatingSystem - if [HPXML::HVACTypeElectricResistance, - HPXML::HVACTypeStove].include? hvac_system.heating_system_type + case hvac_system.heating_system_type + when HPXML::HVACTypeElectricResistance, + HPXML::HVACTypeStove hvac_system.heating_efficiency_percent = 0.1 - elsif [HPXML::HVACTypeFurnace, + when HPXML::HVACTypeFurnace, HPXML::HVACTypeWallFurnace, - HPXML::HVACTypeBoiler].include? hvac_system.heating_system_type + HPXML::HVACTypeBoiler hvac_system.heating_efficiency_afue = 0.1 end elsif hvac_system.is_a? HPXML::CoolingSystem - if [HPXML::HVACTypeCentralAirConditioner].include? hvac_system.cooling_system_type + case hvac_system.cooling_system_type + when HPXML::HVACTypeCentralAirConditioner hvac_system.cooling_efficiency_seer = 0.1 - elsif [HPXML::HVACTypeRoomAirConditioner].include? hvac_system.cooling_system_type + when HPXML::HVACTypeRoomAirConditioner hvac_system.cooling_efficiency_eer = 0.1 end elsif hvac_system.is_a? HPXML::HeatPump - if [HPXML::HVACTypeHeatPumpAirToAir, - HPXML::HVACTypeHeatPumpMiniSplit].include? hvac_system.heat_pump_type + case hvac_system.heat_pump_type + when HPXML::HVACTypeHeatPumpAirToAir, + HPXML::HVACTypeHeatPumpMiniSplit hvac_system.cooling_efficiency_seer = 0.1 hvac_system.heating_efficiency_hspf = 0.1 - elsif [HPXML::HVACTypeHeatPumpGroundToAir].include? hvac_system.heat_pump_type + when HPXML::HVACTypeHeatPumpGroundToAir hvac_system.cooling_efficiency_eer = 0.1 hvac_system.heating_efficiency_cop = 0.1 end end end - elsif ['hvac-setpoints-high'].include? warning_case + when 'hvac-setpoints-high' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_controls[0].heating_setpoint_temp = 100 hpxml_bldg.hvac_controls[0].cooling_setpoint_temp = 100 - elsif ['hvac-setpoints-low'].include? warning_case + when 'hvac-setpoints-low' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_controls[0].heating_setpoint_temp = 0 hpxml_bldg.hvac_controls[0].cooling_setpoint_temp = 0 - elsif ['integrated-heating-efficiency-low'].include? warning_case + when 'integrated-heating-efficiency-low' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ptac-with-heating-electricity.xml') hpxml_bldg.cooling_systems[0].integrated_heating_system_efficiency_percent = 0.4 - elsif ['lighting-groups-missing'].include? warning_case + when 'lighting-groups-missing' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-garage.xml') hpxml_bldg.lighting_groups.reverse_each do |lg| lg.delete end - elsif ['missing-attached-surfaces'].include? warning_case + when 'missing-attached-surfaces' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.building_construction.residential_facility_type = HPXML::ResidentialTypeSFA hpxml_bldg.air_infiltration_measurements[0].infiltration_type = HPXML::InfiltrationTypeUnitExterior - elsif ['hvac-research-features-onoff-thermostat-temperature-capacitance-multiplier-one'].include? warning_case + when 'hvac-research-features-onoff-thermostat-temperature-capacitance-multiplier-one' hpxml, _hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml.header.temperature_capacitance_multiplier = 1 - elsif ['plug-load-type-sauna'].include? warning_case + when 'plug-load-type-sauna' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.plug_loads[0].plug_load_type = HPXML::PlugLoadTypeSauna - elsif ['plug-load-type-aquarium'].include? warning_case + when 'plug-load-type-aquarium' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.plug_loads[0].plug_load_type = HPXML::PlugLoadTypeAquarium - elsif ['plug-load-type-water-bed'].include? warning_case + when 'plug-load-type-water-bed' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.plug_loads[0].plug_load_type = HPXML::PlugLoadTypeWaterBed - elsif ['plug-load-type-space-heater'].include? warning_case + when 'plug-load-type-space-heater' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.plug_loads[0].plug_load_type = HPXML::PlugLoadTypeSpaceHeater - elsif ['plug-load-type-computer'].include? warning_case + when 'plug-load-type-computer' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.plug_loads[0].plug_load_type = HPXML::PlugLoadTypeComputer - elsif ['plug-load-type-tv-crt'].include? warning_case + when 'plug-load-type-tv-crt' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.plug_loads[0].plug_load_type = HPXML::PlugLoadTypeTelevisionCRT - elsif ['plug-load-type-tv-plasma'].include? warning_case + when 'plug-load-type-tv-plasma' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.plug_loads[0].plug_load_type = HPXML::PlugLoadTypeTelevisionPlasma - elsif ['portable-spa'].include? warning_case + when 'portable-spa' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.portable_spas.add(id: 'PorableSpa') - elsif ['slab-zero-exposed-perimeter'].include? warning_case + when 'slab-zero-exposed-perimeter' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.slabs[0].exposed_perimeter = 0 - elsif ['slab-ext-horiz-insul-without-perim-insul'].include? warning_case + when 'slab-ext-horiz-insul-without-perim-insul' hpxml, hpxml_bldg = _create_hpxml('base-foundation-slab-exterior-horizontal-insulation.xml') hpxml_bldg.slabs[0].perimeter_insulation_r_value = 0 - elsif ['slab-large-exposed-perimeter'].include? warning_case + when 'slab-large-exposed-perimeter' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.slabs[0].exposed_perimeter = hpxml_bldg.slabs[0].area * 2 + 1 - elsif ['unit-multiplier'].include? warning_case + when 'unit-multiplier' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.building_construction.number_of_units = 5 - elsif ['window-exterior-shading-types'].include? warning_case + when 'window-exterior-shading-types' hpxml, _hpxml_bldg = _create_hpxml('base-enclosure-windows-shading-types-detailed.xml') - elsif ['wrong-units'].include? warning_case + when 'wrong-units' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-overhangs.xml') hpxml_bldg.slabs[0].thickness = 0.5 hpxml_bldg.foundation_walls[0].thickness = 72.0 @@ -1197,73 +1202,74 @@ def test_ruby_error_messages puts "[#{i + 1}/#{all_expected_errors.size}] Testing #{error_case}..." building_id = nil # Create HPXML object - if ['battery-bad-values-max-greater-than-one'].include? error_case + case error_case + when 'battery-bad-values-max-greater-than-one' hpxml, hpxml_bldg = _create_hpxml('base-battery-scheduled.xml') csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), hpxml_bldg.header.schedules_filepaths[0])) csv_data[1][0] = 1.1 File.write(@tmp_csv_path, csv_data.map(&:to_csv).join) hpxml_bldg.header.schedules_filepaths = [@tmp_csv_path] - elsif ['battery-bad-values-min-less-than-neg-one'].include? error_case + when 'battery-bad-values-min-less-than-neg-one' hpxml, hpxml_bldg = _create_hpxml('base-battery-scheduled.xml') csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), hpxml_bldg.header.schedules_filepaths[0])) csv_data[1][0] = -1.1 File.write(@tmp_csv_path, csv_data.map(&:to_csv).join) hpxml_bldg.header.schedules_filepaths = [@tmp_csv_path] - elsif ['cfis-with-hydronic-distribution'].include? error_case + when 'cfis-with-hydronic-distribution' hpxml, hpxml_bldg = _create_hpxml('base-hvac-boiler-gas-only.xml') hpxml_bldg.ventilation_fans.add(id: "VentilationFan#{hpxml_bldg.ventilation_fans.size + 1}", fan_type: HPXML::MechVentTypeCFIS, used_for_whole_building_ventilation: true, distribution_system_idref: hpxml_bldg.hvac_distributions[0].id) - elsif ['cfis-invalid-supplemental-fan'].include? error_case + when 'cfis-invalid-supplemental-fan' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-cfis-supplemental-fan-exhaust.xml') suppl_fan = hpxml_bldg.ventilation_fans.find { |f| f.is_cfis_supplemental_fan } suppl_fan.fan_type = HPXML::MechVentTypeBalanced - elsif ['cfis-invalid-supplemental-fan2'].include? error_case + when 'cfis-invalid-supplemental-fan2' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-cfis-supplemental-fan-exhaust.xml') suppl_fan = hpxml_bldg.ventilation_fans.find { |f| f.is_cfis_supplemental_fan } suppl_fan.used_for_whole_building_ventilation = false suppl_fan.used_for_garage_ventilation = true - elsif ['cfis-invalid-supplemental-fan3'].include? error_case + when 'cfis-invalid-supplemental-fan3' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-cfis-supplemental-fan-exhaust.xml') suppl_fan = hpxml_bldg.ventilation_fans.find { |f| f.is_cfis_supplemental_fan } suppl_fan.is_shared_system = true suppl_fan.fraction_recirculation = 0.0 suppl_fan.in_unit_flow_rate = suppl_fan.tested_flow_rate / 2.0 - elsif ['cfis-invalid-supplemental-fan4'].include? error_case + when 'cfis-invalid-supplemental-fan4' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-cfis-supplemental-fan-exhaust.xml') suppl_fan = hpxml_bldg.ventilation_fans.find { |f| f.is_cfis_supplemental_fan } suppl_fan.hours_in_operation = 12.0 - elsif ['dehumidifier-setpoints'].include? error_case + when 'dehumidifier-setpoints' hpxml, hpxml_bldg = _create_hpxml('base-appliances-dehumidifier-multiple.xml') hpxml_bldg.dehumidifiers[-1].rh_setpoint = 0.55 - elsif ['desuperheater-with-detailed-setpoints'].include? error_case + when 'desuperheater-with-detailed-setpoints' hpxml, hpxml_bldg = _create_hpxml('base-dhw-tank-detailed-setpoints.xml') hpxml_bldg.water_heating_systems[0].uses_desuperheater = true hpxml_bldg.water_heating_systems[0].related_hvac_idref = hpxml_bldg.cooling_systems[0].id - elsif ['duplicate-id'].include? error_case + when 'duplicate-id' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.plug_loads[-1].id = hpxml_bldg.plug_loads[0].id - elsif ['emissions-duplicate-names'].include? error_case + when 'emissions-duplicate-names' hpxml, _hpxml_bldg = _create_hpxml('base-misc-emissions.xml') hpxml.header.emissions_scenarios << hpxml.header.emissions_scenarios[0].dup - elsif ['emissions-wrong-columns'].include? error_case + when 'emissions-wrong-columns' hpxml, _hpxml_bldg = _create_hpxml('base-misc-emissions.xml') scenario = hpxml.header.emissions_scenarios[1] csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), scenario.elec_schedule_filepath)) csv_data[10] = [431.0] * (scenario.elec_schedule_column_number - 1) File.write(@tmp_csv_path, csv_data.map(&:to_csv).join) hpxml.header.emissions_scenarios[1].elec_schedule_filepath = @tmp_csv_path - elsif ['emissions-wrong-filename'].include? error_case + when 'emissions-wrong-filename' hpxml, _hpxml_bldg = _create_hpxml('base-misc-emissions.xml') hpxml.header.emissions_scenarios[1].elec_schedule_filepath = 'invalid-wrong-filename.csv' - elsif ['emissions-wrong-rows'].include? error_case + when 'emissions-wrong-rows' hpxml, _hpxml_bldg = _create_hpxml('base-misc-emissions.xml') scenario = hpxml.header.emissions_scenarios[1] csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), scenario.elec_schedule_filepath)) File.write(@tmp_csv_path, csv_data[0..-2].map(&:to_csv).join) hpxml.header.emissions_scenarios[1].elec_schedule_filepath = @tmp_csv_path - elsif ['geothermal-loop-multiple-attached-hps'].include? error_case + when 'geothermal-loop-multiple-attached-hps' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump-detailed-geothermal-loop.xml') hpxml_bldg.heat_pumps[0].fraction_cool_load_served = 0.5 hpxml_bldg.heat_pumps[0].fraction_heat_load_served = 0.5 @@ -1276,35 +1282,35 @@ def test_ruby_error_messages annual_cooling_dse: 1.0, annual_heating_dse: 1.0) hpxml_bldg.heat_pumps[1].distribution_system_idref = hpxml_bldg.hvac_distributions[1].id - elsif ['heat-pump-backup-system-load-fraction'].include? error_case + when 'heat-pump-backup-system-load-fraction' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-var-speed-backup-boiler.xml') hpxml_bldg.heating_systems[0].fraction_heat_load_served = 0.5 hpxml_bldg.heat_pumps[0].fraction_heat_load_served = 0.5 - elsif ['heat-pump-switchover-temp-elec-backup'].include? error_case + when 'heat-pump-switchover-temp-elec-backup' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.heat_pumps[0].backup_heating_switchover_temp = 35.0 - elsif ['hvac-cooling-detailed-performance-incomplete-pair'].include? error_case + when 'hvac-cooling-detailed-performance-incomplete-pair' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-var-speed-detailed-performance.xml') hpxml_bldg.heat_pumps[0].cooling_detailed_performance_data[-1].outdoor_temperature -= 1.0 - elsif ['hvac-heating-detailed-performance-incomplete-pair'].include? error_case + when 'hvac-heating-detailed-performance-incomplete-pair' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-var-speed-detailed-performance.xml') hpxml_bldg.heat_pumps[0].heating_detailed_performance_data[-1].outdoor_temperature -= 1.0 - elsif ['heat-pump-lockout-temps-elec-backup'].include? error_case + when 'heat-pump-lockout-temps-elec-backup' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.heat_pumps[0].compressor_lockout_temp = 35.0 hpxml_bldg.heat_pumps[0].backup_heating_lockout_temp = 35.0 - elsif ['hvac-invalid-distribution-system-type'].include? error_case + when 'hvac-invalid-distribution-system-type' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions.add(id: "HVACDistribution#{hpxml_bldg.hvac_distributions.size + 1}", distribution_system_type: HPXML::HVACDistributionTypeHydronic, hydronic_type: HPXML::HydronicTypeBaseboard) hpxml_bldg.heating_systems[-1].distribution_system_idref = hpxml_bldg.hvac_distributions[-1].id - elsif ['hvac-attached-to-uncond-zone'].include? error_case + when 'hvac-attached-to-uncond-zone' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.hvac_systems.each do |hvac_system| hvac_system.attached_to_zone_idref = hpxml_bldg.zones.find { |zone| zone.zone_type != HPXML::ZoneTypeConditioned }.id end - elsif ['hvac-distribution-different-zones'].include? error_case + when 'hvac-distribution-different-zones' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.zones.add(id: 'ConditionedZoneDup', zone_type: HPXML::ZoneTypeConditioned) @@ -1313,45 +1319,45 @@ def test_ruby_error_messages floor_area: hpxml_bldg.zones[0].spaces[0].floor_area) hpxml_bldg.heating_systems[0].attached_to_zone_idref = hpxml_bldg.conditioned_zones[0].id hpxml_bldg.cooling_systems[0].attached_to_zone_idref = hpxml_bldg.conditioned_zones[-1].id - elsif ['hvac-distribution-multiple-attached-cooling'].include? error_case + when 'hvac-distribution-multiple-attached-cooling' hpxml, hpxml_bldg = _create_hpxml('base-hvac-multiple.xml') hpxml_bldg.heat_pumps[0].distribution_system_idref = 'HVACDistribution2' - elsif ['hvac-distribution-multiple-attached-heating'].include? error_case + when 'hvac-distribution-multiple-attached-heating' hpxml, hpxml_bldg = _create_hpxml('base-hvac-multiple.xml') hpxml_bldg.heat_pumps[0].distribution_system_idref = 'HVACDistribution1' - elsif ['hvac-dse-multiple-attached-cooling'].include? error_case + when 'hvac-dse-multiple-attached-cooling' hpxml, hpxml_bldg = _create_hpxml('base-hvac-dse.xml') hpxml_bldg.cooling_systems[0].fraction_cool_load_served = 0.5 hpxml_bldg.cooling_systems << hpxml_bldg.cooling_systems[0].dup hpxml_bldg.cooling_systems[1].id = "CoolingSystem#{hpxml_bldg.cooling_systems.size}" hpxml_bldg.cooling_systems[0].primary_system = false - elsif ['hvac-dse-multiple-attached-heating'].include? error_case + when 'hvac-dse-multiple-attached-heating' hpxml, hpxml_bldg = _create_hpxml('base-hvac-dse.xml') hpxml_bldg.heating_systems[0].fraction_heat_load_served = 0.5 hpxml_bldg.heating_systems << hpxml_bldg.heating_systems[0].dup hpxml_bldg.heating_systems[1].id = "HeatingSystem#{hpxml_bldg.heating_systems.size}" hpxml_bldg.heating_systems[0].primary_system = false - elsif ['hvac-research-features-onoff-thermostat-num-speeds-greater-than-two'].include? error_case + when 'hvac-research-features-onoff-thermostat-num-speeds-greater-than-two' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml_bldg.heat_pumps[0].compressor_type = HPXML::HVACCompressorTypeVariableSpeed - elsif ['hvac-research-features-num-unit-greater-than-one'].include? error_case + when 'hvac-research-features-num-unit-greater-than-one' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml_bldg.building_construction.number_of_units = 2 - elsif ['hvac-gshp-invalid-bore-depth-autosized'].include? error_case + when 'hvac-gshp-invalid-bore-depth-autosized' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump.xml') hpxml_bldg.site.ground_conductivity = 0.1 - elsif ['hvac-gshp-invalid-num-bore-holes'].include? error_case + when 'hvac-gshp-invalid-num-bore-holes' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump-detailed-geothermal-loop.xml') hpxml_bldg.geothermal_loops[0].num_bore_holes = 5 - elsif ['hvac-gshp-invalid-num-bore-holes-autosized'].include? error_case + when 'hvac-gshp-invalid-num-bore-holes-autosized' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump.xml') hpxml_bldg.heat_pumps[0].cooling_capacity *= 2 hpxml_bldg.site.ground_conductivity = 0.08 - elsif ['hvac-inconsistent-fan-powers'].include? error_case + when 'hvac-inconsistent-fan-powers' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.cooling_systems[0].fan_watts_per_cfm = 0.55 hpxml_bldg.heating_systems[0].fan_watts_per_cfm = 0.45 - elsif ['hvac-shared-boiler-multiple'].include? error_case + when 'hvac-shared-boiler-multiple' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-boiler-only-baseboard.xml') hpxml_bldg.hvac_distributions << hpxml_bldg.hvac_distributions[0].dup hpxml_bldg.hvac_distributions[-1].id = "HVACDistribution#{hpxml_bldg.hvac_distributions.size}" @@ -1361,7 +1367,7 @@ def test_ruby_error_messages hpxml_bldg.heating_systems[1].id = "HeatingSystem#{hpxml_bldg.heating_systems.size}" hpxml_bldg.heating_systems[1].distribution_system_idref = hpxml_bldg.hvac_distributions[-1].id hpxml_bldg.heating_systems[1].primary_system = true - elsif ['hvac-shared-chiller-multiple'].include? error_case + when 'hvac-shared-chiller-multiple' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-chiller-only-baseboard.xml') hpxml_bldg.hvac_distributions << hpxml_bldg.hvac_distributions[0].dup hpxml_bldg.hvac_distributions[-1].id = "HVACDistribution#{hpxml_bldg.hvac_distributions.size}" @@ -1371,214 +1377,214 @@ def test_ruby_error_messages hpxml_bldg.cooling_systems[1].id = "CoolingSystem#{hpxml_bldg.cooling_systems.size}" hpxml_bldg.cooling_systems[1].distribution_system_idref = hpxml_bldg.hvac_distributions[-1].id hpxml_bldg.cooling_systems[1].primary_system = true - elsif ['hvac-shared-chiller-negative-seer-eq'].include? error_case + when 'hvac-shared-chiller-negative-seer-eq' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-chiller-only-baseboard.xml') hpxml_bldg.cooling_systems[0].shared_loop_watts *= 100.0 - elsif ['inconsistent-belly-wing-skirt-present'].include? error_case + when 'inconsistent-belly-wing-skirt-present' hpxml, hpxml_bldg = _create_hpxml('base-foundation-belly-wing-skirt.xml') fnd = hpxml_bldg.foundations.find { |f| f.foundation_type == HPXML::FoundationTypeBellyAndWing } hpxml_bldg.foundations << fnd.dup hpxml_bldg.foundations[-1].id = 'Duplicate' hpxml_bldg.foundations[-1].belly_wing_skirt_present = false - elsif ['inconsistent-cond-zone-assignment'].include? error_case + when 'inconsistent-cond-zone-assignment' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') grg_ceiling = hpxml_bldg.floors.find { |f| f.interior_adjacent_to == HPXML::LocationGarage && f.exterior_adjacent_to == HPXML::LocationAtticUnvented } grg_ceiling.attached_to_space_idref = hpxml_bldg.conditioned_spaces[0].id - elsif ['inconsistent-uncond-basement-within-infiltration-volume'].include? error_case + when 'inconsistent-uncond-basement-within-infiltration-volume' hpxml, hpxml_bldg = _create_hpxml('base-foundation-unconditioned-basement.xml') fnd = hpxml_bldg.foundations.find { |f| f.foundation_type == HPXML::FoundationTypeBasementUnconditioned } hpxml_bldg.foundations << fnd.dup hpxml_bldg.foundations[-1].id = 'Duplicate' hpxml_bldg.foundations[-1].within_infiltration_volume = true - elsif ['inconsistent-unvented-attic-within-infiltration-volume'].include? error_case + when 'inconsistent-unvented-attic-within-infiltration-volume' hpxml, hpxml_bldg = _create_hpxml('base.xml') attic = hpxml_bldg.attics.find { |a| a.attic_type == HPXML::AtticTypeUnvented } hpxml_bldg.attics << attic.dup hpxml_bldg.attics[-1].id = 'Duplicate' hpxml_bldg.attics[-1].within_infiltration_volume = true - elsif ['inconsistent-unvented-crawl-within-infiltration-volume'].include? error_case + when 'inconsistent-unvented-crawl-within-infiltration-volume' hpxml, hpxml_bldg = _create_hpxml('base-foundation-unvented-crawlspace.xml') fnd = hpxml_bldg.foundations.find { |f| f.foundation_type == HPXML::FoundationTypeCrawlspaceUnvented } hpxml_bldg.foundations << fnd.dup hpxml_bldg.foundations[-1].id = 'Duplicate' hpxml_bldg.foundations[-1].within_infiltration_volume = true - elsif ['inconsistent-vented-attic-ventilation-rate'].include? error_case + when 'inconsistent-vented-attic-ventilation-rate' hpxml, hpxml_bldg = _create_hpxml('base-atticroof-vented.xml') attic = hpxml_bldg.attics.find { |a| a.attic_type == HPXML::AtticTypeVented } hpxml_bldg.attics << attic.dup hpxml_bldg.attics[-1].id = 'Duplicate' hpxml_bldg.attics[-1].vented_attic_sla *= 2 - elsif ['inconsistent-vented-attic-ventilation-rate2'].include? error_case + when 'inconsistent-vented-attic-ventilation-rate2' hpxml, hpxml_bldg = _create_hpxml('base-atticroof-vented.xml') attic = hpxml_bldg.attics.find { |a| a.attic_type == HPXML::AtticTypeVented } hpxml_bldg.attics << attic.dup hpxml_bldg.attics[-1].id = 'Duplicate' hpxml_bldg.attics[-1].vented_attic_sla = nil hpxml_bldg.attics[-1].vented_attic_ach = 5.0 - elsif ['inconsistent-vented-crawl-ventilation-rate'].include? error_case + when 'inconsistent-vented-crawl-ventilation-rate' hpxml, hpxml_bldg = _create_hpxml('base-foundation-vented-crawlspace.xml') fnd = hpxml_bldg.foundations.find { |f| f.foundation_type == HPXML::FoundationTypeCrawlspaceVented } hpxml_bldg.foundations << fnd.dup hpxml_bldg.foundations[-1].id = 'Duplicate' hpxml_bldg.foundations[-1].vented_crawlspace_sla *= 2 - elsif ['invalid-battery-capacity-units'].include? error_case + when 'invalid-battery-capacity-units' hpxml, hpxml_bldg = _create_hpxml('base-pv-battery.xml') hpxml_bldg.batteries[0].usable_capacity_kwh = nil hpxml_bldg.batteries[0].usable_capacity_ah = 200.0 - elsif ['invalid-battery-capacity-units2'].include? error_case + when 'invalid-battery-capacity-units2' hpxml, hpxml_bldg = _create_hpxml('base-pv-battery-ah.xml') hpxml_bldg.batteries[0].usable_capacity_kwh = 10.0 hpxml_bldg.batteries[0].usable_capacity_ah = nil - elsif ['invalid-datatype-boolean'].include? error_case + when 'invalid-datatype-boolean' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.roofs[0].radiant_barrier = false - elsif ['invalid-datatype-integer'].include? error_case + when 'invalid-datatype-integer' hpxml, _hpxml_bldg = _create_hpxml('base.xml') - elsif ['invalid-datatype-float'].include? error_case + when 'invalid-datatype-float' hpxml, _hpxml_bldg = _create_hpxml('base-misc-emissions.xml') - elsif ['invalid-daylight-saving'].include? error_case + when 'invalid-daylight-saving' hpxml, hpxml_bldg = _create_hpxml('base-simcontrol-daylight-saving-custom.xml') hpxml_bldg.dst_begin_month = 3 hpxml_bldg.dst_begin_day = 10 hpxml_bldg.dst_end_month = 4 hpxml_bldg.dst_end_day = 31 - elsif ['invalid-distribution-cfa-served'].include? error_case + when 'invalid-distribution-cfa-served' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[-1].conditioned_floor_area_served = 2701.1 - elsif ['invalid-epw-filepath'].include? error_case + when 'invalid-epw-filepath' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.climate_and_risk_zones.weather_station_epw_filepath = 'foo.epw' - elsif ['invalid-holiday-lighting-dates'].include? error_case + when 'invalid-holiday-lighting-dates' hpxml, hpxml_bldg = _create_hpxml('base-lighting-holiday.xml') hpxml_bldg.lighting.holiday_period_begin_month = 11 hpxml_bldg.lighting.holiday_period_begin_day = 31 hpxml_bldg.lighting.holiday_period_end_month = 1 hpxml_bldg.lighting.holiday_period_end_day = 15 - elsif ['invalid-id'].include? error_case + when 'invalid-id' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-skylights.xml') hpxml_bldg.skylights[0].id = '' - elsif ['invalid-neighbor-shading-azimuth'].include? error_case + when 'invalid-neighbor-shading-azimuth' hpxml, hpxml_bldg = _create_hpxml('base-misc-neighbor-shading.xml') hpxml_bldg.neighbor_buildings[0].azimuth = 145 - elsif ['invalid-ptac-dse'].include? error_case + when 'invalid-ptac-dse' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ptac-cfis.xml') hpxml_bldg.hvac_distributions[0].annual_cooling_dse = 0.9 - elsif ['invalid-pthp-dse'].include? error_case + when 'invalid-pthp-dse' hpxml, hpxml_bldg = _create_hpxml('base-hvac-pthp-cfis.xml') hpxml_bldg.hvac_distributions[0].annual_heating_dse = 0.9 - elsif ['invalid-relatedhvac-dhw-indirect'].include? error_case + when 'invalid-relatedhvac-dhw-indirect' hpxml, hpxml_bldg = _create_hpxml('base-dhw-indirect.xml') hpxml_bldg.water_heating_systems[0].related_hvac_idref = 'HeatingSystem_bad' - elsif ['invalid-relatedhvac-desuperheater'].include? error_case + when 'invalid-relatedhvac-desuperheater' hpxml, hpxml_bldg = _create_hpxml('base-hvac-central-ac-only-1-speed.xml') hpxml_bldg.water_heating_systems[0].uses_desuperheater = true hpxml_bldg.water_heating_systems[0].related_hvac_idref = 'CoolingSystem_bad' - elsif ['invalid-runperiod'].include? error_case + when 'invalid-runperiod' hpxml, _hpxml_bldg = _create_hpxml('base.xml') hpxml.header.sim_begin_month = 3 hpxml.header.sim_begin_day = 10 hpxml.header.sim_end_month = 4 hpxml.header.sim_end_day = 31 - elsif ['invalid-shading-season'].include? error_case + when 'invalid-shading-season' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.header.shading_summer_begin_month = 3 hpxml_bldg.header.shading_summer_begin_day = 10 hpxml_bldg.header.shading_summer_end_month = 4 hpxml_bldg.header.shading_summer_end_day = 31 - elsif ['invalid-unavailable-period'].include? error_case + when 'invalid-unavailable-period' hpxml, _hpxml_bldg = _create_hpxml('base.xml') hpxml.header.unavailable_periods.add(column_name: 'Power Outage', begin_month: 3, begin_day: 10, end_month: 4, end_day: 31) - elsif ['invalid-schema-version'].include? error_case + when 'invalid-schema-version' hpxml, _hpxml_bldg = _create_hpxml('base.xml') - elsif ['invalid-skylights-physical-properties'].include? error_case + when 'invalid-skylights-physical-properties' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-skylights-physical-properties.xml') hpxml_bldg.skylights[1].thermal_break = false - elsif ['invalid-windows-physical-properties'].include? error_case + when 'invalid-windows-physical-properties' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-windows-physical-properties.xml') hpxml_bldg.windows[2].thermal_break = false - elsif ['inverter-unequal-efficiencies'].include? error_case + when 'inverter-unequal-efficiencies' hpxml, hpxml_bldg = _create_hpxml('base-pv.xml') hpxml_bldg.inverters.add(id: 'Inverter2', inverter_efficiency: 0.5) hpxml_bldg.pv_systems[1].inverter_idref = hpxml_bldg.inverters[-1].id - elsif ['leap-year-TMY'].include? error_case + when 'leap-year-TMY' hpxml, _hpxml_bldg = _create_hpxml('base-simcontrol-calendar-year-custom.xml') hpxml.header.sim_calendar_year = 2008 - elsif ['net-area-negative-roof-floor'].include? error_case + when 'net-area-negative-roof-floor' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-skylights.xml') hpxml_bldg.skylights[0].area = 4000 - elsif ['net-area-negative-wall'].include? error_case + when 'net-area-negative-wall' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.windows[0].area = 1000 - elsif ['orphaned-geothermal-loop'].include? error_case + when 'orphaned-geothermal-loop' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump-detailed-geothermal-loop.xml') hpxml_bldg.heat_pumps[0].geothermal_loop_idref = nil - elsif ['orphaned-hvac-distribution'].include? error_case + when 'orphaned-hvac-distribution' hpxml, hpxml_bldg = _create_hpxml('base-hvac-furnace-gas-room-ac.xml') hpxml_bldg.heating_systems[0].delete hpxml_bldg.hvac_controls[0].heating_setpoint_temp = nil - elsif ['refrigerators-multiple-primary'].include? error_case + when 'refrigerators-multiple-primary' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml_bldg.refrigerators[1].primary_indicator = true - elsif ['refrigerators-no-primary'].include? error_case + when 'refrigerators-no-primary' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml_bldg.refrigerators[0].primary_indicator = false - elsif ['repeated-relatedhvac-dhw-indirect'].include? error_case + when 'repeated-relatedhvac-dhw-indirect' hpxml, hpxml_bldg = _create_hpxml('base-dhw-indirect.xml') hpxml_bldg.water_heating_systems[0].fraction_dhw_load_served = 0.5 hpxml_bldg.water_heating_systems << hpxml_bldg.water_heating_systems[0].dup hpxml_bldg.water_heating_systems[1].id = "WaterHeatingSystem#{hpxml_bldg.water_heating_systems.size}" - elsif ['repeated-relatedhvac-desuperheater'].include? error_case + when 'repeated-relatedhvac-desuperheater' hpxml, hpxml_bldg = _create_hpxml('base-hvac-central-ac-only-1-speed.xml') hpxml_bldg.water_heating_systems[0].fraction_dhw_load_served = 0.5 hpxml_bldg.water_heating_systems[0].uses_desuperheater = true hpxml_bldg.water_heating_systems[0].related_hvac_idref = 'CoolingSystem1' hpxml_bldg.water_heating_systems << hpxml_bldg.water_heating_systems[0].dup hpxml_bldg.water_heating_systems[1].id = "WaterHeatingSystem#{hpxml_bldg.water_heating_systems.size}" - elsif ['schedule-detailed-bad-values-max-not-one'].include? error_case + when 'schedule-detailed-bad-values-max-not-one' hpxml, hpxml_bldg = _create_hpxml('base-schedules-detailed-occupancy-stochastic.xml') csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), hpxml_bldg.header.schedules_filepaths[0])) csv_data[1][1] = 1.1 File.write(@tmp_csv_path, csv_data.map(&:to_csv).join) hpxml_bldg.header.schedules_filepaths = [@tmp_csv_path] - elsif ['schedule-detailed-bad-values-negative'].include? error_case + when 'schedule-detailed-bad-values-negative' hpxml, hpxml_bldg = _create_hpxml('base-schedules-detailed-occupancy-stochastic.xml') csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), hpxml_bldg.header.schedules_filepaths[0])) csv_data[1][1] = -0.5 File.write(@tmp_csv_path, csv_data.map(&:to_csv).join) hpxml_bldg.header.schedules_filepaths = [@tmp_csv_path] - elsif ['schedule-detailed-bad-values-non-numeric'].include? error_case + when 'schedule-detailed-bad-values-non-numeric' hpxml, hpxml_bldg = _create_hpxml('base-schedules-detailed-occupancy-stochastic.xml') csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), hpxml_bldg.header.schedules_filepaths[0])) csv_data[1][1] = 'NA' File.write(@tmp_csv_path, csv_data.map(&:to_csv).join) hpxml_bldg.header.schedules_filepaths = [@tmp_csv_path] - elsif ['schedule-detailed-bad-values-mode-negative'].include? error_case + when 'schedule-detailed-bad-values-mode-negative' hpxml, hpxml_bldg = _create_hpxml('base-dhw-tank-heat-pump-detailed-schedules.xml') csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), hpxml_bldg.header.schedules_filepaths[1])) csv_data[1][0] = -0.5 File.write(@tmp_csv_path, csv_data.map(&:to_csv).join) hpxml_bldg.header.schedules_filepaths = [@tmp_csv_path] - elsif ['schedule-detailed-duplicate-columns'].include? error_case + when 'schedule-detailed-duplicate-columns' hpxml, hpxml_bldg = _create_hpxml('base-schedules-detailed-occupancy-stochastic.xml') csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), hpxml_bldg.header.schedules_filepaths[0])) File.write(@tmp_csv_path, csv_data.map(&:to_csv).join) hpxml_bldg.header.schedules_filepaths = [] hpxml_bldg.header.schedules_filepaths << @tmp_csv_path hpxml_bldg.header.schedules_filepaths << @tmp_csv_path - elsif ['schedule-detailed-wrong-filename'].include? error_case + when 'schedule-detailed-wrong-filename' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.header.schedules_filepaths << 'invalid-wrong-filename.csv' - elsif ['schedule-detailed-wrong-rows'].include? error_case + when 'schedule-detailed-wrong-rows' hpxml, hpxml_bldg = _create_hpxml('base-schedules-detailed-occupancy-stochastic.xml') csv_data = CSV.read(File.join(File.dirname(hpxml.hpxml_path), hpxml_bldg.header.schedules_filepaths[0])) File.write(@tmp_csv_path, csv_data[0..-2].map(&:to_csv).join) hpxml_bldg.header.schedules_filepaths = [@tmp_csv_path] - elsif ['skylight-not-connected-to-cond-space'].include? error_case + when 'skylight-not-connected-to-cond-space' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-garage.xml') hpxml_bldg.skylights.add(id: 'Skylight1', area: 15.0, @@ -1589,7 +1595,7 @@ def test_ruby_error_messages shaft_assembly_r_value: 6.25, attached_to_roof_idref: 'Roof1', attached_to_floor_idref: 'Floor1') - elsif ['solar-thermal-system-with-combi-tankless'].include? error_case + when 'solar-thermal-system-with-combi-tankless' hpxml, hpxml_bldg = _create_hpxml('base-dhw-combi-tankless.xml') hpxml_bldg.solar_thermal_systems.add(id: "SolarThermalSystem#{hpxml_bldg.solar_thermal_systems.size + 1}", system_type: HPXML::SolarThermalSystemTypeHotWater, @@ -1601,7 +1607,7 @@ def test_ruby_error_messages collector_rated_optical_efficiency: 0.77, collector_rated_thermal_losses: 0.793, water_heating_system_idref: 'WaterHeatingSystem1') - elsif ['solar-thermal-system-with-desuperheater'].include? error_case + when 'solar-thermal-system-with-desuperheater' hpxml, hpxml_bldg = _create_hpxml('base-dhw-desuperheater.xml') hpxml_bldg.solar_thermal_systems.add(id: "SolarThermalSystem#{hpxml_bldg.solar_thermal_systems.size + 1}", system_type: HPXML::SolarThermalSystemTypeHotWater, @@ -1613,7 +1619,7 @@ def test_ruby_error_messages collector_rated_optical_efficiency: 0.77, collector_rated_thermal_losses: 0.793, water_heating_system_idref: 'WaterHeatingSystem1') - elsif ['solar-thermal-system-with-dhw-indirect'].include? error_case + when 'solar-thermal-system-with-dhw-indirect' hpxml, hpxml_bldg = _create_hpxml('base-dhw-combi-tankless.xml') hpxml_bldg.solar_thermal_systems.add(id: "SolarThermalSystem#{hpxml_bldg.solar_thermal_systems.size + 1}", system_type: HPXML::SolarThermalSystemTypeHotWater, @@ -1625,88 +1631,88 @@ def test_ruby_error_messages collector_rated_optical_efficiency: 0.77, collector_rated_thermal_losses: 0.793, water_heating_system_idref: 'WaterHeatingSystem1') - elsif ['storm-windows-unexpected-window-ufactor'].include? error_case + when 'storm-windows-unexpected-window-ufactor' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.windows[0].storm_type = 'clear' - elsif ['surface-attached-to-uncond-space'].include? error_case + when 'surface-attached-to-uncond-space' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.walls[-1].attached_to_space_idref = hpxml_bldg.zones.find { |zone| zone.zone_type != HPXML::ZoneTypeConditioned }.spaces[0].id - elsif ['surface-attached-to-uncond-space2'].include? error_case + when 'surface-attached-to-uncond-space2' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.slabs[-1].attached_to_space_idref = hpxml_bldg.zones.find { |zone| zone.zone_type != HPXML::ZoneTypeConditioned }.spaces[0].id - elsif ['unattached-cfis'].include? error_case + when 'unattached-cfis' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.ventilation_fans.add(id: "VentilationFan#{hpxml_bldg.ventilation_fans.size + 1}", fan_type: HPXML::MechVentTypeCFIS, used_for_whole_building_ventilation: true, distribution_system_idref: hpxml_bldg.hvac_distributions[0].id) hpxml_bldg.ventilation_fans[0].distribution_system_idref = 'foobar' - elsif ['unattached-door'].include? error_case + when 'unattached-door' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.doors[0].attached_to_wall_idref = 'foobar' - elsif ['unattached-gshp'].include? error_case + when 'unattached-gshp' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump-detailed-geothermal-loop.xml') hpxml_bldg.heat_pumps[0].geothermal_loop_idref = 'foobar' hpxml_bldg.geothermal_loops[0].delete - elsif ['unattached-hvac-distribution'].include? error_case + when 'unattached-hvac-distribution' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.heating_systems[0].distribution_system_idref = 'foobar' - elsif ['unattached-pv-system'].include? error_case + when 'unattached-pv-system' hpxml, hpxml_bldg = _create_hpxml('base-pv.xml') hpxml_bldg.pv_systems[0].inverter_idref = 'foobar' - elsif ['unattached-skylight'].include? error_case + when 'unattached-skylight' hpxml, hpxml_bldg = _create_hpxml('base-enclosure-skylights.xml') hpxml_bldg.skylights[0].attached_to_roof_idref = 'foobar' hpxml_bldg.skylights[0].attached_to_floor_idref = 'foobar' - elsif ['unattached-solar-thermal-system'].include? error_case + when 'unattached-solar-thermal-system' hpxml, hpxml_bldg = _create_hpxml('base-dhw-solar-indirect-flat-plate.xml') hpxml_bldg.solar_thermal_systems[0].water_heating_system_idref = 'foobar' - elsif ['unattached-shared-clothes-washer-dhw-distribution'].include? error_case + when 'unattached-shared-clothes-washer-dhw-distribution' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-laundry-room.xml') hpxml_bldg.clothes_washers[0].water_heating_system_idref = nil hpxml_bldg.clothes_washers[0].hot_water_distribution_idref = 'foobar' - elsif ['unattached-shared-clothes-washer-water-heater'].include? error_case + when 'unattached-shared-clothes-washer-water-heater' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-laundry-room.xml') hpxml_bldg.clothes_washers[0].water_heating_system_idref = 'foobar' - elsif ['unattached-shared-dishwasher-dhw-distribution'].include? error_case + when 'unattached-shared-dishwasher-dhw-distribution' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-laundry-room.xml') hpxml_bldg.dishwashers[0].water_heating_system_idref = nil hpxml_bldg.dishwashers[0].hot_water_distribution_idref = 'foobar' - elsif ['unattached-shared-dishwasher-water-heater'].include? error_case + when 'unattached-shared-dishwasher-water-heater' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-unit-shared-laundry-room.xml') hpxml_bldg.dishwashers[0].water_heating_system_idref = 'foobar' - elsif ['unattached-window'].include? error_case + when 'unattached-window' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.windows[0].attached_to_wall_idref = 'foobar' - elsif ['unattached-zone'].include? error_case + when 'unattached-zone' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.heating_systems[0].attached_to_zone_idref = 'foobar' hpxml_bldg.cooling_systems[0].attached_to_zone_idref = 'foobar' - elsif ['unavailable-period-missing-column'].include? error_case + when 'unavailable-period-missing-column' hpxml, _hpxml_bldg = _create_hpxml('base-schedules-simple-vacancy.xml') hpxml.header.unavailable_periods[0].column_name = 'foobar' - elsif ['unique-objects-vary-across-units-epw'].include? error_case + when 'unique-objects-vary-across-units-epw' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-whole-building.xml', building_id: building_id) hpxml_bldg.climate_and_risk_zones.weather_station_epw_filepath = 'USA_AZ_Phoenix-Sky.Harbor.Intl.AP.722780_TMY3.epw' - elsif ['unique-objects-vary-across-units-dst'].include? error_case + when 'unique-objects-vary-across-units-dst' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-whole-building.xml', building_id: building_id) hpxml_bldg.dst_begin_month = 3 hpxml_bldg.dst_begin_day = 15 hpxml_bldg.dst_end_month = 10 hpxml_bldg.dst_end_day = 15 - elsif ['unique-objects-vary-across-units-tmains'].include? error_case + when 'unique-objects-vary-across-units-tmains' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-whole-building.xml', building_id: building_id) hpxml_bldg.hot_water_distributions[0].dwhr_facilities_connected = HPXML::DWHRFacilitiesConnectedOne hpxml_bldg.hot_water_distributions[0].dwhr_equal_flow = true hpxml_bldg.hot_water_distributions[0].dwhr_efficiency = 0.55 - elsif ['whole-mf-building-batteries'].include? error_case + when 'whole-mf-building-batteries' hpxml, hpxml_bldg = _create_hpxml('base-bldgtype-mf-whole-building.xml', building_id: building_id) hpxml_bldg.batteries.add(id: 'Battery1', type: HPXML::BatteryTypeLithiumIon) - elsif ['whole-mf-building-dehumidifiers-unit-multiplier'].include? error_case + when 'whole-mf-building-dehumidifiers-unit-multiplier' hpxml, hpxml_bldg = _create_hpxml('base-appliances-dehumidifier.xml') hpxml_bldg.building_construction.number_of_units = 2 - elsif ['whole-mf-building-gshps-unit-multiplier'].include? error_case + when 'whole-mf-building-gshps-unit-multiplier' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump.xml') hpxml_bldg.building_construction.number_of_units = 2 else @@ -1716,13 +1722,14 @@ def test_ruby_error_messages hpxml_doc = hpxml.to_doc() # Perform additional raw XML manipulation - if ['invalid-datatype-boolean'].include? error_case + case error_case + when 'invalid-datatype-boolean' XMLHelper.get_element(hpxml_doc, '/HPXML/Building/BuildingDetails/Enclosure/Roofs/Roof/RadiantBarrier').inner_text = 'FOOBAR' - elsif ['invalid-datatype-integer'].include? error_case + when 'invalid-datatype-integer' XMLHelper.get_element(hpxml_doc, '/HPXML/Building/BuildingDetails/BuildingSummary/BuildingConstruction/NumberofBedrooms').inner_text = '2.5' - elsif ['invalid-datatype-float'].include? error_case + when 'invalid-datatype-float' XMLHelper.get_element(hpxml_doc, '/HPXML/SoftwareInfo/extension/EmissionsScenarios/EmissionsScenario/EmissionsFactor/Value').inner_text = 'FOOBAR' - elsif ['invalid-schema-version'].include? error_case + when 'invalid-schema-version' root = XMLHelper.get_element(hpxml_doc, '/HPXML') XMLHelper.add_attribute(root, 'schemaVersion', '2.3') end @@ -1847,11 +1854,12 @@ def test_ruby_warning_messages puts "[#{i + 1}/#{all_expected_warnings.size}] Testing #{warning_case}..." building_id = nil # Create HPXML object - if ['cfis-undersized-supplemental-fan'].include? warning_case + case warning_case + when 'cfis-undersized-supplemental-fan' hpxml, hpxml_bldg = _create_hpxml('base-mechvent-cfis-supplemental-fan-exhaust.xml') suppl_fan = hpxml_bldg.ventilation_fans.find { |f| f.is_cfis_supplemental_fan } suppl_fan.tested_flow_rate = 90.0 - elsif ['duct-lto-cfm25-cond-space'].include? warning_case + when 'duct-lto-cfm25-cond-space' hpxml, hpxml_bldg = _create_hpxml('base-atticroof-conditioned.xml') hpxml_bldg.hvac_distributions[0].conditioned_floor_area_served = hpxml_bldg.building_construction.conditioned_floor_area hpxml_bldg.hvac_distributions[0].duct_leakage_measurements.each do |dlm| @@ -1862,12 +1870,12 @@ def test_ruby_warning_messages duct.duct_surface_area = nil duct.duct_location = nil end - elsif ['duct-lto-cfm25-uncond-space'].include? warning_case + when 'duct-lto-cfm25-uncond-space' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_distributions[0].duct_leakage_measurements.each do |dlm| dlm.duct_leakage_value = 800.0 end - elsif ['duct-lto-cfm50-cond-space'].include? warning_case + when 'duct-lto-cfm50-cond-space' hpxml, hpxml_bldg = _create_hpxml('base-atticroof-conditioned.xml') hpxml_bldg.hvac_distributions[0].conditioned_floor_area_served = hpxml_bldg.building_construction.conditioned_floor_area hpxml_bldg.hvac_distributions[0].duct_leakage_measurements.each do |dlm| @@ -1878,12 +1886,12 @@ def test_ruby_warning_messages duct.duct_surface_area = nil duct.duct_location = nil end - elsif ['duct-lto-cfm50-uncond-space'].include? warning_case + when 'duct-lto-cfm50-uncond-space' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ducts-leakage-cfm50.xml') hpxml_bldg.hvac_distributions[0].duct_leakage_measurements.each do |dlm| dlm.duct_leakage_value = 1600.0 end - elsif ['duct-lto-percent-cond-space'].include? warning_case + when 'duct-lto-percent-cond-space' hpxml, hpxml_bldg = _create_hpxml('base-atticroof-conditioned.xml') hpxml_bldg.hvac_distributions[0].conditioned_floor_area_served = hpxml_bldg.building_construction.conditioned_floor_area hpxml_bldg.hvac_distributions[0].duct_leakage_measurements.each do |dlm| @@ -1894,59 +1902,59 @@ def test_ruby_warning_messages duct.duct_surface_area = nil duct.duct_location = nil end - elsif ['duct-lto-percent-uncond-space'].include? warning_case + when 'duct-lto-percent-uncond-space' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ducts-leakage-percent.xml') hpxml_bldg.hvac_distributions[0].duct_leakage_measurements.each do |dlm| dlm.duct_leakage_value = 0.25 end - elsif ['floor-or-ceiling1'].include? warning_case + when 'floor-or-ceiling1' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.floors[0].floor_or_ceiling = HPXML::FloorOrCeilingFloor - elsif ['floor-or-ceiling2'].include? warning_case + when 'floor-or-ceiling2' hpxml, hpxml_bldg = _create_hpxml('base-foundation-unvented-crawlspace.xml') hpxml_bldg.floors[0].floor_or_ceiling = HPXML::FloorOrCeilingCeiling - elsif ['hvac-gshp-bore-depth-autosized-high'].include? warning_case + when 'hvac-gshp-bore-depth-autosized-high' hpxml, hpxml_bldg = _create_hpxml('base-hvac-ground-to-air-heat-pump.xml') hpxml_bldg.site.ground_conductivity = 0.07 - elsif ['hvac-seasons'].include? warning_case + when 'hvac-seasons' hpxml, hpxml_bldg = _create_hpxml('base-hvac-seasons.xml') - elsif ['hvac-setpoint-adjustments'].include? warning_case + when 'hvac-setpoint-adjustments' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.hvac_controls[0].heating_setpoint_temp = 76.0 hpxml_bldg.hvac_controls[0].cooling_setpoint_temp = 75.0 - elsif ['hvac-setpoint-adjustments-daily-setbacks'].include? warning_case + when 'hvac-setpoint-adjustments-daily-setbacks' hpxml, hpxml_bldg = _create_hpxml('base-hvac-setpoints-daily-setbacks.xml') hpxml_bldg.hvac_controls[0].heating_setback_temp = 76.0 hpxml_bldg.hvac_controls[0].cooling_setpoint_temp = 75.0 - elsif ['hvac-setpoint-adjustments-daily-schedules'].include? warning_case + when 'hvac-setpoint-adjustments-daily-schedules' hpxml, hpxml_bldg = _create_hpxml('base-hvac-setpoints-daily-schedules.xml') hpxml_bldg.hvac_controls[0].weekday_heating_setpoints = '64, 64, 64, 64, 64, 64, 64, 76, 70, 66, 66, 66, 66, 66, 66, 66, 66, 68, 68, 68, 68, 68, 64, 64' - elsif ['manualj-sum-space-num-occupants'].include? warning_case + when 'manualj-sum-space-num-occupants' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.header.manualj_num_occupants = 4.8 hpxml_bldg.conditioned_spaces.each_with_index do |space, i| space.manualj_num_occupants = (i == 0 ? hpxml_bldg.header.manualj_num_occupants.round : 0) end - elsif ['manualj-sum-space-internal-loads-sensible'].include? warning_case + when 'manualj-sum-space-internal-loads-sensible' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.header.manualj_internal_loads_sensible = 1000.0 hpxml_bldg.conditioned_spaces.each_with_index do |space, i| space.manualj_internal_loads_sensible = (i == 0 ? 1200.0 : 0) end - elsif ['manualj-sum-space-internal-loads-latent'].include? warning_case + when 'manualj-sum-space-internal-loads-latent' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces.xml') hpxml_bldg.header.manualj_internal_loads_latent = 200.0 hpxml_bldg.conditioned_spaces.each_with_index do |space, i| space.manualj_internal_loads_latent = (i == 0 ? 100.0 : 0) end - elsif ['multiple-conditioned-zone'].include? warning_case + when 'multiple-conditioned-zone' hpxml, hpxml_bldg = _create_hpxml('base-zones-spaces-multiple.xml') - elsif ['power-outage'].include? warning_case + when 'power-outage' hpxml, _hpxml_bldg = _create_hpxml('base-schedules-simple-power-outage.xml') - elsif ['multistage-backup-more-than-4-stages'].include? warning_case + when 'multistage-backup-more-than-4-stages' hpxml, _hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed-research-features.xml') hpxml.header.heat_pump_backup_heating_capacity_increment = 5000 - elsif ['schedule-file-and-weekday-weekend-multipliers'].include? warning_case + when 'schedule-file-and-weekday-weekend-multipliers' hpxml, hpxml_bldg = _create_hpxml('base-misc-loads-large-uncommon.xml') hpxml.header.utility_bill_scenarios.clear # we don't want the propane warning hpxml_bldg.header.schedules_filepaths << File.join(File.dirname(__FILE__), '../resources/schedule_files/occupancy-stochastic.csv') @@ -1956,7 +1964,7 @@ def test_ruby_warning_messages hpxml_bldg.hot_water_distributions[0].recirculation_pump_weekday_fractions = @default_schedules_csv_data["#{SchedulesFile::Columns[:HotWaterRecirculationPump].name}_no_control"]['RecirculationPumpWeekdayScheduleFractions'] hpxml_bldg.hot_water_distributions[0].recirculation_pump_weekend_fractions = @default_schedules_csv_data["#{SchedulesFile::Columns[:HotWaterRecirculationPump].name}_no_control"]['RecirculationPumpWeekendScheduleFractions'] hpxml_bldg.hot_water_distributions[0].recirculation_pump_monthly_multipliers = @default_schedules_csv_data[SchedulesFile::Columns[:HotWaterRecirculationPump].name]['RecirculationPumpMonthlyScheduleMultipliers'] - elsif ['schedule-file-and-refrigerators-freezer-coefficients'].include? warning_case + when 'schedule-file-and-refrigerators-freezer-coefficients' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.header.schedules_filepaths << File.join(File.dirname(__FILE__), '../resources/schedule_files/occupancy-stochastic.csv') hpxml_bldg.header.schedules_filepaths << File.join(File.dirname(__FILE__), '../resources/schedule_files/occupancy-non-stochastic.csv') @@ -1969,20 +1977,20 @@ def test_ruby_warning_messages hpxml_bldg.freezers.add(id: "Freezer#{hpxml_bldg.freezers.size + 1}", constant_coefficients: '-0.487, -0.340, -0.370, -0.361, -0.515, -0.684, -0.471, -0.159, -0.079, -0.417, -0.411, -0.386, -0.240, -0.314, -0.160, -0.121, -0.469, -0.412, -0.091, 0.077, -0.118, -0.247, -0.445, -0.544', temperature_coefficients: '0.019, 0.016, 0.017, 0.016, 0.018, 0.021, 0.019, 0.015, 0.015, 0.019, 0.018, 0.018, 0.016, 0.017, 0.015, 0.015, 0.020, 0.020, 0.017, 0.014, 0.016, 0.017, 0.019, 0.020') - elsif ['schedule-file-and-setpoints'].include? warning_case + when 'schedule-file-and-setpoints' hpxml, hpxml_bldg = _create_hpxml('base.xml') hpxml_bldg.header.schedules_filepaths << File.join(File.dirname(__FILE__), '../resources/schedule_files/setpoints.csv') hpxml_bldg.header.schedules_filepaths << File.join(File.dirname(__FILE__), '../resources/schedule_files/water-heater-setpoints.csv') - elsif ['schedule-file-and-operating-mode'].include? warning_case + when 'schedule-file-and-operating-mode' hpxml, hpxml_bldg = _create_hpxml('base-dhw-tank-heat-pump-operating-mode-heat-pump-only.xml') hpxml_bldg.header.schedules_filepaths << File.join(File.dirname(__FILE__), '../resources/schedule_files/water-heater-operating-modes.csv') - elsif ['schedule-file-max-power-ratio-with-single-speed-system'].include? warning_case + when 'schedule-file-max-power-ratio-with-single-speed-system' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-1-speed.xml') hpxml_bldg.header.schedules_filepaths << File.join(File.dirname(__FILE__), '../resources/schedule_files/hvac-variable-system-maximum-power-ratios-varied.csv') - elsif ['schedule-file-max-power-ratio-with-two-speed-system'].include? warning_case + when 'schedule-file-max-power-ratio-with-two-speed-system' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-2-speed.xml') hpxml_bldg.header.schedules_filepaths << File.join(File.dirname(__FILE__), '../resources/schedule_files/hvac-variable-system-maximum-power-ratios-varied.csv') - elsif ['schedule-file-max-power-ratio-with-separate-backup-system'].include? warning_case + when 'schedule-file-max-power-ratio-with-separate-backup-system' hpxml, hpxml_bldg = _create_hpxml('base-hvac-air-to-air-heat-pump-var-speed-backup-boiler.xml') hpxml_bldg.header.schedules_filepaths << File.join(File.dirname(__FILE__), '../resources/schedule_files/hvac-variable-system-maximum-power-ratios-varied.csv') else diff --git a/README.md b/README.md index d5f9c473ea..900da4515b 100644 --- a/README.md +++ b/README.md @@ -4,7 +4,7 @@ [![ci](https://github.com/NREL/OpenStudio-HPXML/workflows/ci/badge.svg)](https://github.com/NREL/OpenStudio-HPXML/actions) [![Documentation Status](https://readthedocs.org/projects/openstudio-hpxml/badge/?version=latest)](https://openstudio-hpxml.readthedocs.io/en/latest/?badge=latest) -OpenStudio-HPXML allows running residential EnergyPlusā„¢ simulations using an [HPXML file](https://hpxml.nrel.gov/) for the building description. +OpenStudio-HPXML allows running residential [EnergyPlusā„¢ simulations](https://energyplus.net/) using an [HPXML file](https://hpxml.nrel.gov/) for the building description. It is intended to be used by user interfaces or other automated software workflows that automatically produce the HPXML file. OpenStudio-HPXML can accommodate a wide range of different building technologies and geometries. @@ -31,6 +31,8 @@ OpenStudio-HPXML capabilities include: - Annual and timeseries outputs (energy, loads, temperatures, etc.) - Optional HPXML inputs with transparent defaults - Schematron and XSD Schema input validation +- Can be used for [DOE HOMES program approval](https://www.energy.gov/scep/single-family-modeling-solutions-home-efficiency-rebates-program) +- Can be used for [ACCA Manual J approval](https://www.acca.org/standards/approved-software) ## Measures diff --git a/ReportSimulationOutput/measure.rb b/ReportSimulationOutput/measure.rb index c342ba06c0..fcec01b1bd 100644 --- a/ReportSimulationOutput/measure.rb +++ b/ReportSimulationOutput/measure.rb @@ -657,13 +657,14 @@ def get_timestamps(msgpackData, hpxml_header, hpxml_bldgs, args) # Convert from EnergyPlus default (end-of-timestep) to start-of-timestep convention if args[:timeseries_timestamp_convention] == 'start' - if args[:timeseries_frequency] == 'timestep' + case args[:timeseries_frequency] + when 'timestep' ts_offset = hpxml_header.timestep * 60 # seconds - elsif args[:timeseries_frequency] == 'hourly' + when 'hourly' ts_offset = 60 * 60 # seconds - elsif args[:timeseries_frequency] == 'daily' + when 'daily' ts_offset = 60 * 60 * 24 # seconds - elsif args[:timeseries_frequency] == 'monthly' + when 'monthly' ts_offset = Calendar.num_days_in_months(year)[month - 1] * 60 * 60 * 24 # seconds else fail "Unexpected timeseries_frequency: #{args[:timeseries_frequency]}." @@ -1836,13 +1837,14 @@ def report_timeseries_output_results(runner, outputs, timeseries_output_path, ar year = @hpxml_header.sim_calendar_year start_day = Calendar.get_day_num_from_month_day(year, @hpxml_header.sim_begin_month, @hpxml_header.sim_begin_day) start_hr = (start_day - 1) * 24 - if args[:timeseries_frequency] == 'timestep' + case args[:timeseries_frequency] + when 'timestep' interval_hrs = @hpxml_header.timestep / 60.0 - elsif args[:timeseries_frequency] == 'hourly' + when 'hourly' interval_hrs = 1.0 - elsif args[:timeseries_frequency] == 'daily' + when 'daily' interval_hrs = 24.0 - elsif args[:timeseries_frequency] == 'monthly' + when 'monthly' interval_hrs = Calendar.num_days_in_year(year) * 24.0 / 12 end header_data = [['wxDVFileHeaderVer.1'], diff --git a/ReportSimulationOutput/measure.xml b/ReportSimulationOutput/measure.xml index 92e37b1986..3d43e4ede3 100644 --- a/ReportSimulationOutput/measure.xml +++ b/ReportSimulationOutput/measure.xml @@ -3,8 +3,8 @@ 3.1 report_simulation_output df9d170c-c21a-4130-866d-0d46b06073fd - 443dae63-bc28-4988-8cb0-5fc10c452c68 - 2024-10-13T19:48:30Z + 5e609cb1-8e83-479b-afda-c1de10cd62ea + 2024-11-27T02:33:41Z 9BF1E6AC ReportSimulationOutput HPXML Simulation Output Report @@ -1929,7 +1929,7 @@ measure.rb rb script - 45BFC7E3 + 4C7478A8 test_report_sim_output.rb diff --git a/ReportUtilityBills/measure.rb b/ReportUtilityBills/measure.rb index e2819c9420..a4e6158b79 100644 --- a/ReportUtilityBills/measure.rb +++ b/ReportUtilityBills/measure.rb @@ -199,8 +199,7 @@ def energyPlusOutputRequests(runner, user_arguments) FT::Coal => HPXML::FuelTypeCoal } # Check for presence of fuels once - has_fuel = hpxml.has_fuels(hpxml.to_doc) - has_fuel[HPXML::FuelTypeElectricity] = true + has_fuel = hpxml.has_fuels() # Has production has_pv = @hpxml_buildings.count { |hpxml_bldg| !hpxml_bldg.pv_systems.empty? } > 0 @@ -543,7 +542,8 @@ def get_utility_rates(hpxml_path, fuels, utility_rates, bill_scenario, pv_monthl utility_rates.each do |fuel_type, rate| next if fuels[[fuel_type, false]].timeseries.sum == 0 - if fuel_type == FT::Elec + case fuel_type + when FT::Elec if bill_scenario.elec_tariff_filepath.nil? rate.fixed_charge_monthly = bill_scenario.elec_fixed_charge rate.flat_rate = bill_scenario.elec_marginal_rate @@ -618,22 +618,22 @@ def get_utility_rates(hpxml_path, fuels, utility_rates, bill_scenario, pv_monthl # Feed-In Tariff rate.feed_in_tariff_rate = bill_scenario.pv_feed_in_tariff_rate if bill_scenario.pv_compensation_type == HPXML::PVCompensationTypeFeedInTariff - elsif fuel_type == FT::Gas + when FT::Gas rate.fixed_charge_monthly = bill_scenario.natural_gas_fixed_charge rate.flat_rate = bill_scenario.natural_gas_marginal_rate - elsif fuel_type == FT::Oil + when FT::Oil rate.fixed_charge_monthly = bill_scenario.fuel_oil_fixed_charge rate.flat_rate = bill_scenario.fuel_oil_marginal_rate - elsif fuel_type == FT::Propane + when FT::Propane rate.fixed_charge_monthly = bill_scenario.propane_fixed_charge rate.flat_rate = bill_scenario.propane_marginal_rate - elsif fuel_type == FT::WoodCord + when FT::WoodCord rate.fixed_charge_monthly = bill_scenario.wood_fixed_charge rate.flat_rate = bill_scenario.wood_marginal_rate - elsif fuel_type == FT::WoodPellets + when FT::WoodPellets rate.fixed_charge_monthly = bill_scenario.wood_pellets_fixed_charge rate.flat_rate = bill_scenario.wood_pellets_marginal_rate - elsif fuel_type == FT::Coal + when FT::Coal rate.fixed_charge_monthly = bill_scenario.coal_fixed_charge rate.flat_rate = bill_scenario.coal_marginal_rate end diff --git a/ReportUtilityBills/measure.xml b/ReportUtilityBills/measure.xml index cb75deec1e..eaca6d9b0e 100644 --- a/ReportUtilityBills/measure.xml +++ b/ReportUtilityBills/measure.xml @@ -3,8 +3,8 @@ 3.1 report_utility_bills ca88a425-e59a-4bc4-af51-c7e7d1e960fe - 6b3c51ee-095b-455c-b1f1-a1cb47e3b4a8 - 2024-11-14T00:26:10Z + 3d7e2753-fab2-4d89-bfd9-208a59523248 + 2024-11-27T02:33:42Z 15BF4E57 ReportUtilityBills Utility Bills Report @@ -180,7 +180,7 @@ measure.rb rb script - 4D23174A + 48B476E1 detailed_rates/README.md @@ -360,7 +360,7 @@ test_report_utility_bills.rb rb test - 1D59A593 + 5B90EFCE diff --git a/ReportUtilityBills/tests/test_report_utility_bills.rb b/ReportUtilityBills/tests/test_report_utility_bills.rb index b95a848dd5..84406d63e1 100644 --- a/ReportUtilityBills/tests/test_report_utility_bills.rb +++ b/ReportUtilityBills/tests/test_report_utility_bills.rb @@ -79,7 +79,7 @@ def setup fuel_oil_marginal_rate: 3.495346153846154) # Check for presence of fuels once - has_fuel = @hpxml_bldg.has_fuels(@hpxml.to_doc) + has_fuel = @hpxml_bldg.has_fuels() Defaults.apply_header(@hpxml_header, @hpxml_bldg, nil) Defaults.apply_utility_bill_scenarios(nil, @hpxml_header, @hpxml_bldg, has_fuel) @@ -1176,15 +1176,16 @@ def _load_timeseries(pv_size_kw, use_hourly_electricity) values = col[1..-1].map { |v| Float(v) } - if col_name == 'Electricity [kWh]' + case col_name + when 'Electricity [kWh]' fuels[[FT::Elec, false]].timeseries = values - elsif col_name == 'Gas [therm]' + when 'Gas [therm]' fuels[[FT::Gas, false]].timeseries = values - elsif col_name == 'Propane [gal]' + when 'Propane [gal]' fuels[[FT::Propane, false]].timeseries = values - elsif col_name == 'Oil [gal]' + when 'Oil [gal]' fuels[[FT::Oil, false]].timeseries = values - elsif col_name == "PV_#{pv_size_kw}kW [kWh]" + when "PV_#{pv_size_kw}kW [kWh]" fuels[[FT::Elec, true]].timeseries = values end end diff --git a/docs/source/images/slab.png b/docs/source/images/slab.png index 347cc4ee6647d30b556f61f16a95544fbb3da778..367be073bedcab2892e37559ddb74ce6684a8815 100644 GIT binary patch literal 51931 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 KP;XRqME)OGi^e4Y literal 150528 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

-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 aW^R$kbfdS!)srOrqjga4K$gboYySgQn-fz2 diff --git a/docs/source/intro.rst b/docs/source/intro.rst index 8ffc2be114..9fff0c5d17 100644 --- a/docs/source/intro.rst +++ b/docs/source/intro.rst @@ -20,6 +20,14 @@ OpenStudio-HPXML capabilities include: - Optional HPXML inputs with transparent defaults - Schematron and XSD Schema input validation +.. note:: + If you are seeking `DOE HOMES program approval `_, OpenStudio-HPXML can be used to meet the `energy modeling software tests `_. + +.. note:: + + If you are seeking `ACCA Manual J approval `_ for your software, you will need to contact ACCA and go through their approval process. + OpenStudio-HPXML design load calculations can be used to obtain approval, but additional Manual J-specific HPXML inputs need to be specified; refer to the HPXML test files at ``workflow/tests/ACCA_Examples``. + Accuracy vs Speed ----------------- diff --git a/docs/source/testing_framework.rst b/docs/source/testing_framework.rst index 103845eb43..d6625cd8ea 100644 --- a/docs/source/testing_framework.rst +++ b/docs/source/testing_framework.rst @@ -10,8 +10,7 @@ The current set of tests include: - RESNETĀ® HERSĀ® HVAC tests - RESNET HERS DSE tests - RESNET HERS Hot Water tests - -OpenStudio-HPXML can be used to meet BPI-2400 software tests. +- ACCA Manual J tests Running Tests Locally --------------------- diff --git a/docs/source/workflow_inputs.rst b/docs/source/workflow_inputs.rst index 56e2b4ad2a..4b996f0805 100644 --- a/docs/source/workflow_inputs.rst +++ b/docs/source/workflow_inputs.rst @@ -594,7 +594,7 @@ Building occupancy is entered in ``/HPXML/Building/BuildingDetails/BuildingSumma \- **single-family attached**: NumberofBedrooms = -1.98 + 1.89 * NumberofResidents - \- **apartment unit or multifamily**: NumberofBedrooms = -1.36 + 1.49 * NumberofResidents + \- **apartment unit**: NumberofBedrooms = -1.36 + 1.49 * NumberofResidents .. [#] If WeekdayScheduleFractions or WeekendScheduleFractions not provided (and :ref:`schedules_detailed` not used), then :ref:`schedules_default` are used. .. [#] If MonthlyScheduleMultipliers not provided (and :ref:`schedules_detailed` not used), then :ref:`schedules_default` are used. @@ -3523,12 +3523,12 @@ If using Manual J default duct factor tables, additional information is entered ================================ ======= ============ =========== ======== ========= ========================================================= Element Type Units Constraints Required Default Notes ================================ ======= ============ =========== ======== ========= ========================================================= - `TableNumber` string See [#]_ Yes Manual J Default Duct Factor Table number - `LookupFloorArea` double ft2 > 0 Yes Lookup floor area value for the Manual J table [#]_ - `LeakageLevel` string See [#]_ Yes Leakage tightness value for the Manual J table - `InsulationRValue` double F-ft2-hr/Btu >= 2 Yes Insulation R-value for the Manual J table - `SupplySurfaceArea` or `DSF` double ft2 or frac >= 0 No DSF=1 Surface area or estimated fraction of supply ducts in unconditioned space - `ReturnSurfaceArea` or `DSF` double ft2 or frac >= 0 No DSF=1 Surface area or estimated fraction of return ducts in unconditioned space + ``TableNumber`` string See [#]_ Yes Manual J Default Duct Factor Table number + ``LookupFloorArea`` double ft2 > 0 Yes Lookup floor area value for the Manual J table [#]_ + ``LeakageLevel`` string See [#]_ Yes Leakage tightness value for the Manual J table + ``InsulationRValue`` double F-ft2-hr/Btu >= 2 Yes Insulation R-value for the Manual J table + ``SupplySurfaceArea`` or ``DSF`` double ft2 or frac >= 0 No DSF=1 Surface area or estimated fraction of supply ducts in unconditioned space + ``ReturnSurfaceArea`` or ``DSF`` double ft2 or frac >= 0 No DSF=1 Surface area or estimated fraction of return ducts in unconditioned space ================================ ======= ============ =========== ======== ========= ========================================================= .. [#] TableNumber choices are "7A-R", "7A-T", "7B-R", "7B-T", "7A-AE", "7B-AE", "7C-AE", "7C-R", "7C-T", "7D-R", "7D-T", "7E-R", "7E-T", "7F-R", "7F-T", "7G-R", "7G-T", "7H", "7I", "7D-AE", "7J-1", "7J-2", "7K", "7L", "7M", "7N", "7O-1", "7O-2", "7O-3", "7O-4", "7P-1", "7P-2", "7P-3", or "7P-4". diff --git a/tasks.rb b/tasks.rb index a80bc6614a..008d2ef4ee 100644 --- a/tasks.rb +++ b/tasks.rb @@ -1656,18 +1656,6 @@ def apply_hpxml_modification_sample_files(hpxml_path, hpxml) hpxml_bldg.hvac_distributions[0].ducts[2].duct_surface_area = 37.5 hpxml_bldg.hvac_distributions[0].ducts[3].duct_location = HPXML::LocationConditionedSpace hpxml_bldg.hvac_distributions[0].ducts[3].duct_surface_area = 12.5 - if hpxml_file == 'base-hvac-ducts-area-fractions.xml' - hpxml_bldg.hvac_distributions[0].ducts[0].duct_surface_area = nil - hpxml_bldg.hvac_distributions[0].ducts[1].duct_surface_area = nil - hpxml_bldg.hvac_distributions[0].ducts[2].duct_surface_area = nil - hpxml_bldg.hvac_distributions[0].ducts[3].duct_surface_area = nil - hpxml_bldg.hvac_distributions[0].ducts[0].duct_fraction_area = 0.75 - hpxml_bldg.hvac_distributions[0].ducts[1].duct_fraction_area = 0.75 - hpxml_bldg.hvac_distributions[0].ducts[2].duct_fraction_area = 0.25 - hpxml_bldg.hvac_distributions[0].ducts[3].duct_fraction_area = 0.25 - hpxml_bldg.hvac_distributions[0].conditioned_floor_area_served = 4050.0 - hpxml_bldg.hvac_distributions[0].number_of_return_registers = 3 - end elsif ['base-hvac-ducts-effective-rvalue.xml'].include? hpxml_file hpxml_bldg.hvac_distributions[0].ducts[0].duct_insulation_r_value = nil hpxml_bldg.hvac_distributions[0].ducts[1].duct_insulation_r_value = nil @@ -1953,13 +1941,6 @@ def apply_hpxml_modification_sample_files(hpxml_path, hpxml) if hpxml_file.include? 'base-hvac-ground-to-air-heat-pump-detailed-geothermal-loop.xml' hpxml_bldg.geothermal_loops[0].shank_spacing = 2.5 end - if hpxml_file.include? 'HERS_HVAC' - hpxml_bldg.hvac_distributions.clear - hpxml_bldg.hvac_distributions.add(id: "HVACDistribution#{hpxml_bldg.hvac_distributions.size + 1}", - distribution_system_type: HPXML::HVACDistributionTypeDSE, - annual_heating_dse: 1.0, - annual_cooling_dse: 1.0) - end hpxml_bldg.heating_systems.each do |heating_system| if heating_system.heating_system_type == HPXML::HVACTypeBoiler && heating_system.heating_system_fuel == HPXML::FuelTypeNaturalGas && diff --git a/workflow/hpxml_inputs.json b/workflow/hpxml_inputs.json index 443c200788..0d0bdac382 100644 --- a/workflow/hpxml_inputs.json +++ b/workflow/hpxml_inputs.json @@ -3699,10 +3699,6 @@ "parent_hpxml": "sample_files/base.xml", "geometry_unit_num_occupants": 0 }, - "sample_files/base-residents-0-runperiod-1-month.xml": { - "parent_hpxml": "sample_files/base-residents-0.xml", - "simulation_control_run_period": "Feb 1 - Feb 28" - }, "sample_files/base-residents-1.xml": { "parent_hpxml": "sample_files/base.xml", "geometry_unit_num_occupants": 1, diff --git a/workflow/sample_files/base-location-detailed.xml b/workflow/sample_files/base-location-detailed.xml index e53ed3915f..6bf985d616 100644 --- a/workflow/sample_files/base-location-detailed.xml +++ b/workflow/sample_files/base-location-detailed.xml @@ -62,6 +62,12 @@ 21600.0 + + + 2006 + 5B + + diff --git a/workflow/sample_files/base-residents-0-runperiod-1-month.xml b/workflow/sample_files/base-residents-0-runperiod-1-month.xml deleted file mode 100644 index 0c03c64524..0000000000 --- a/workflow/sample_files/base-residents-0-runperiod-1-month.xml +++ /dev/null @@ -1,558 +0,0 @@ - - - - HPXML - tasks.rb - 2000-01-01T00:00:00-07:00 - create - - - - - 60 - 2 - 1 - 2 - 28 - - - - Bills - - - - - - - - -

- CO -
- - - proposed workscope - - - - - suburban - stand-alone - no units above or below - 180 - - electricity - natural gas - - - - 0.0 - - - single-family detached - 2.0 - 1.0 - 8.0 - 3 - 2 - 2700.0 - 21600.0 - - - - - 2006 - 5B - - - - USA_CO_Denver.Intl.AP.725650_TMY3 - - USA_CO_Denver.Intl.AP.725650_TMY3.epw - - - - - - - - 50.0 - - ACH - 3.0 - - 21600.0 - - - - - - - - false - - - false - - - - - - - - - - - true - - - - - - - - - - - attic - unvented - 1509.3 - asphalt or fiberglass shingles - 0.7 - 0.92 - 6.0 - - - 2.3 - - - - - - - outside - basement - conditioned - 115.6 - wood siding - 0.7 - 0.92 - - - 23.0 - - - - - - - outside - conditioned space - - - - 1200.0 - wood siding - 0.7 - 0.92 - - gypsum board - - - - 23.0 - - - - - outside - attic - unvented - gable - - - - 225.0 - wood siding - 0.7 - 0.92 - - - 4.0 - - - - - - - ground - basement - conditioned - 8.0 - 1200.0 - 8.0 - 7.0 - - gypsum board - - - - - continuous - exterior - 8.9 - 0.0 - 8.0 - - - continuous - interior - 0.0 - - - - - - - - attic - unvented - conditioned space - ceiling - - - - 1350.0 - - gypsum board - - - - 39.3 - - - - - - - basement - conditioned - 1350.0 - 4.0 - 150.0 - - - - 0.0 - 0.0 - - - - - - 0.0 - 0.0 - - - - 0.0 - 0.0 - - - - - - - 108.0 - 0 - 0.33 - 0.45 - - - 0.7 - 0.85 - - 0.67 - - - - - 72.0 - 90 - 0.33 - 0.45 - - - 0.7 - 0.85 - - 0.67 - - - - - 108.0 - 180 - 0.33 - 0.45 - - - 0.7 - 0.85 - - 0.67 - - - - - 72.0 - 270 - 0.33 - 0.45 - - - 0.7 - 0.85 - - 0.67 - - - - - - - - 40.0 - 180 - 4.4 - - - - - - - - - - - - - - - - - natural gas - 36000.0 - - AFUE - 0.92 - - 1.0 - - - - - central air conditioner - electricity - 24000.0 - single stage - 1.0 - - SEER - 13.0 - - 0.73 - - - - - 68.0 - 78.0 - - - - - - regular velocity - - supply - - CFM25 - 75.0 - to outside - - - - return - - CFM25 - 25.0 - to outside - - - - - supply - 4.0 - attic - unvented - 150.0 - - - - return - 0.0 - attic - unvented - 50.0 - - - - - - - - - electricity - storage water heater - conditioned space - 40.0 - 1.0 - 18767.0 - 0.95 - 125.0 - - - - - - 50.0 - - - - 0.0 - - - - - shower head - true - - - - faucet - false - - - - - - - conditioned space - 1.21 - 380.0 - 0.12 - 1.09 - 27.0 - 6.0 - 3.2 - - - - conditioned space - electricity - 3.73 - true - 150.0 - - - - conditioned space - 307.0 - 12 - 0.12 - 1.09 - 22.32 - 4.0 - - - - conditioned space - 650.0 - - - - conditioned space - electricity - false - - - - false - - - - - - interior - 0.4 - - - - - - - interior - 0.1 - - - - - - - interior - 0.25 - - - - - - - exterior - 0.4 - - - - - - - exterior - 0.1 - - - - - - - exterior - 0.25 - - - - - - - - - TV other - - kWh/year - 620.0 - - - - - other - - kWh/year - 2457.0 - - - 0.855 - 0.045 - - - - - - \ No newline at end of file diff --git a/workflow/tests/ACCA_Examples/Bob_Ross_Residence_3-22.xml b/workflow/tests/ACCA_Examples/Bob_Ross_Residence_3-22.xml index 10640b97b2..47546477c9 100644 --- a/workflow/tests/ACCA_Examples/Bob_Ross_Residence_3-22.xml +++ b/workflow/tests/ACCA_Examples/Bob_Ross_Residence_3-22.xml @@ -552,10 +552,10 @@ garage basement - conditioned concrete block - 8.0 - 260.0 + 6.0 + 156.0 east - 8.0 + 6.0 1.71 diff --git a/workflow/tests/ACCA_Examples/Bob_Ross_Residence_3-23.xml b/workflow/tests/ACCA_Examples/Bob_Ross_Residence_3-23.xml index 01170e03ac..46ce008de5 100644 --- a/workflow/tests/ACCA_Examples/Bob_Ross_Residence_3-23.xml +++ b/workflow/tests/ACCA_Examples/Bob_Ross_Residence_3-23.xml @@ -357,10 +357,9 @@ ground crawlspace - unvented concrete block solid core - 10.0 - 1740.0 - north - 8.0 + 2.0 + 348.0 + 0.0 7.3 @@ -371,10 +370,10 @@ garage crawlspace - unvented concrete block solid core - 10.0 - 260.0 + 2.0 + 52.0 east - 8.0 + 0.0 7.3 diff --git a/workflow/tests/base_results/results_acca_hvac.csv b/workflow/tests/base_results/results_acca_hvac.csv index ab0a9a96d0..5ce0c8096b 100644 --- a/workflow/tests/base_results/results_acca_hvac.csv +++ b/workflow/tests/base_results/results_acca_hvac.csv @@ -15,8 +15,8 @@ Bob_Ross_Residence_3-19.xml,15.0,93.0,48732.0,32721.0,0.0,48732.0,5392.0,7478.0, Bob_Ross_Residence_3-2.xml,15.0,93.0,41962.0,30966.0,0.0,41962.0,4321.0,7478.0,1615.0,808.0,18183.0,0.0,0.0,2454.0,4196.0,0.0,2908.0,0.0,25911.0,2965.0,5849.0,2945.0,470.0,4163.0,0.0,0.0,0.0,3280.0,0.0,952.0,3552.0,1707.0,27.0,2872.0,565.0,0.0,1307.0,1000.0 Bob_Ross_Residence_3-20.xml,15.0,93.0,48009.0,32360.0,0.0,48009.0,4668.0,7478.0,1615.0,808.0,18183.0,0.0,0.0,2454.0,4196.0,2791.0,5815.0,0.0,27310.0,3022.0,5810.0,2945.0,470.0,4163.0,0.0,0.0,0.0,3280.0,457.0,1903.0,3552.0,1707.0,0.0,4807.0,565.0,627.0,2614.0,1000.0 Bob_Ross_Residence_3-21.xml,15.0,93.0,44547.0,31210.0,0.0,44547.0,4929.0,7478.0,1615.0,808.0,18183.0,0.0,0.0,2454.0,4196.0,2791.0,2093.0,0.0,26323.0,3139.0,5810.0,2945.0,470.0,4163.0,0.0,0.0,0.0,3280.0,457.0,799.0,3552.0,1707.0,0.0,4807.0,565.0,627.0,2614.0,1000.0 -Bob_Ross_Residence_3-22.xml,15.0,93.0,55479.0,34561.0,0.0,55479.0,5817.0,7478.0,1615.0,808.0,29949.0,0.0,0.0,2709.0,4196.0,0.0,2908.0,0.0,28660.0,3316.0,5810.0,2945.0,470.0,6628.0,0.0,0.0,0.0,3280.0,0.0,952.0,3552.0,1707.0,0.0,2872.0,565.0,0.0,1307.0,1000.0 -Bob_Ross_Residence_3-23.xml,15.0,93.0,39373.0,32746.0,0.0,39373.0,9948.0,5597.0,1615.0,808.0,10079.0,0.0,4222.0,0.0,4196.0,0.0,2908.0,0.0,26128.0,4490.0,4616.0,2945.0,470.0,3191.0,0.0,1382.0,0.0,3280.0,0.0,952.0,3095.0,1707.0,0.0,6450.0,4539.0,0.0,1307.0,604.0 +Bob_Ross_Residence_3-22.xml,15.0,93.0,54948.0,34561.0,0.0,54948.0,5758.0,7478.0,1615.0,808.0,29422.0,0.0,0.0,2765.0,4196.0,0.0,2908.0,0.0,28660.0,3316.0,5810.0,2945.0,470.0,6628.0,0.0,0.0,0.0,3280.0,0.0,952.0,3552.0,1707.0,0.0,2872.0,565.0,0.0,1307.0,1000.0 +Bob_Ross_Residence_3-23.xml,15.0,93.0,39253.0,32708.0,0.0,39253.0,9916.0,5597.0,1615.0,808.0,10079.0,0.0,4135.0,0.0,4196.0,0.0,2908.0,0.0,26093.0,4484.0,4616.0,2945.0,470.0,3191.0,0.0,1353.0,0.0,3280.0,0.0,952.0,3095.0,1707.0,0.0,6450.0,4539.0,0.0,1307.0,604.0 Bob_Ross_Residence_3-3.xml,15.0,93.0,41962.0,32754.0,0.0,41962.0,4321.0,7478.0,1615.0,808.0,18183.0,0.0,0.0,2454.0,4196.0,0.0,2908.0,0.0,27279.0,3140.0,6752.0,2945.0,470.0,4163.0,0.0,0.0,0.0,3280.0,0.0,952.0,3552.0,1707.0,317.0,2872.0,565.0,0.0,1307.0,1000.0 Bob_Ross_Residence_3-4.xml,15.0,93.0,41962.0,29729.0,0.0,41962.0,4321.0,7478.0,1615.0,808.0,18183.0,0.0,0.0,2454.0,4196.0,0.0,2908.0,0.0,24965.0,2845.0,5051.0,2945.0,470.0,4163.0,0.0,0.0,0.0,3280.0,0.0,952.0,3552.0,1707.0,0.0,2872.0,565.0,0.0,1307.0,1000.0 Bob_Ross_Residence_3-5.xml,15.0,93.0,41962.0,36796.0,0.0,41962.0,4321.0,7478.0,1615.0,808.0,18183.0,0.0,0.0,2454.0,4196.0,0.0,2908.0,0.0,30369.0,3534.0,9693.0,2945.0,470.0,4163.0,0.0,0.0,0.0,3280.0,0.0,952.0,3552.0,1707.0,73.0,2872.0,565.0,0.0,1307.0,1000.0 diff --git a/workflow/tests/base_results/results_simulations_bills.csv b/workflow/tests/base_results/results_simulations_bills.csv index efd5712afc..342ba17740 100644 --- a/workflow/tests/base_results/results_simulations_bills.csv +++ b/workflow/tests/base_results/results_simulations_bills.csv @@ -411,7 +411,6 @@ base-pv-generators-battery-scheduled.xml,955.78,144.0,1068.19,-978.65,233.54,144 base-pv-generators-battery.xml,924.45,144.0,1036.86,-978.65,202.21,144.0,338.28,482.28,0.0,239.96,239.96,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-pv-generators.xml,892.64,144.0,1005.05,-978.65,170.4,144.0,338.28,482.28,0.0,239.96,239.96,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-pv.xml,861.79,144.0,1303.13,-978.65,468.48,144.0,249.31,393.31,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 -base-residents-0-runperiod-1-month.xml,130.03,12.0,18.35,0.0,30.35,12.0,87.68,99.68,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-residents-0.xml,915.08,144.0,268.02,0.0,412.02,144.0,359.06,503.06,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-residents-1-misc-loads-large-uncommon.xml,2674.4,144.0,1817.12,0.0,1961.12,144.0,443.44,587.44,0.0,0.0,0.0,0.0,67.32,67.32,0.0,58.52,58.52,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-residents-1-misc-loads-large-uncommon2.xml,2404.07,144.0,1731.6,0.0,1875.6,144.0,254.61,398.61,0.0,71.34,71.34,0.0,0.0,0.0,0.0,0.0,0.0,0.0,58.52,58.52,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 diff --git a/workflow/tests/base_results/results_simulations_energy.csv b/workflow/tests/base_results/results_simulations_energy.csv index f231923ff7..c33851605c 100644 --- a/workflow/tests/base_results/results_simulations_energy.csv +++ b/workflow/tests/base_results/results_simulations_energy.csv @@ -411,7 +411,6 @@ base-pv-generators-battery-scheduled.xml,78.352,43.276,37.536,2.46,32.316,8.5,0. base-pv-generators-battery.xml,77.491,42.415,36.675,1.599,32.316,8.5,0.0,0.0,0.0,0.0,0.0,0.591,0.0,0.0,4.398,0.662,9.013,0.0,0.0,4.507,0.0,0.334,0.0,0.0,0.0,0.0,2.072,0.0,0.0,0.319,0.365,1.513,1.529,0.0,2.116,8.384,0.0,0.0,0.0,0.0,0.0,0.0,-26.886,-8.189,0.874,23.816,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,8.5,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,8.5,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-pv-generators.xml,76.617,41.541,35.801,0.725,32.316,8.5,0.0,0.0,0.0,0.0,0.0,0.591,0.0,0.0,4.398,0.662,9.013,0.0,0.0,4.507,0.0,0.334,0.0,0.0,0.0,0.0,2.072,0.0,0.0,0.319,0.365,1.513,1.529,0.0,2.116,8.384,0.0,0.0,0.0,0.0,0.0,0.0,-26.886,-8.189,0.0,23.816,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,8.5,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,8.5,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-pv.xml,59.617,32.73,35.801,8.914,23.816,0.0,0.0,0.0,0.0,0.0,0.0,0.591,0.0,0.0,4.398,0.662,9.013,0.0,0.0,4.507,0.0,0.334,0.0,0.0,0.0,0.0,2.072,0.0,0.0,0.319,0.365,1.513,1.529,0.0,2.116,8.384,0.0,0.0,0.0,0.0,0.0,0.0,-26.886,0.0,0.0,23.816,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 -base-residents-0-runperiod-1-month.xml,8.8797,8.8797,0.504,0.504,8.3757,0.0,0.0,0.0,0.0,0.0,0.0,0.2078,0.0,0.0,0.1034,0.0,0.0468,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.146,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,8.3757,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-residents-0.xml,41.664,41.664,7.363,7.363,34.301,0.0,0.0,0.0,0.0,0.0,0.0,0.851,0.0,0.0,3.398,0.481,0.577,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,2.056,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,34.301,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-residents-1-misc-loads-large-uncommon.xml,99.944,99.944,49.922,49.922,42.362,0.0,2.527,5.133,0.0,0.0,0.0,0.565,0.0,0.0,4.562,0.695,3.75,0.0,0.0,4.507,0.0,0.334,0.0,0.0,0.0,0.0,4.293,1.024,0.0,0.2,0.221,0.917,1.115,0.0,2.007,6.55,5.687,1.15,0.0,6.508,2.937,2.899,0.0,0.0,0.0,22.778,0.0,0.0,0.0,0.0,0.0,18.039,0.0,0.0,1.544,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,2.527,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,5.133,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-residents-1-misc-loads-large-uncommon2.xml,79.555,79.555,47.572,47.572,24.322,2.527,0.0,0.0,5.133,0.0,0.0,0.565,0.0,0.0,4.562,0.695,3.75,0.0,0.0,4.507,0.0,0.334,0.0,0.0,0.0,0.0,4.293,1.024,0.0,0.2,0.221,0.917,1.115,0.0,2.007,6.55,5.687,1.15,0.0,6.508,0.587,2.899,0.0,0.0,0.0,22.778,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,1.544,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,2.527,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,5.133,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 diff --git a/workflow/tests/base_results/results_simulations_hvac.csv b/workflow/tests/base_results/results_simulations_hvac.csv index 0377e4c151..d85488c244 100644 --- a/workflow/tests/base_results/results_simulations_hvac.csv +++ b/workflow/tests/base_results/results_simulations_hvac.csv @@ -411,7 +411,6 @@ base-pv-generators-battery-scheduled.xml,6.8,91.76,36000.0,24000.0,0.0,32239.0,8 base-pv-generators-battery.xml,6.8,91.76,36000.0,24000.0,0.0,32239.0,8709.0,7508.0,0.0,575.0,6918.0,0.0,0.0,1738.0,2171.0,4620.0,0.0,0.0,20039.0,6112.0,7037.0,0.0,207.0,448.0,0.0,0.0,0.0,2293.0,622.0,0.0,3320.0,0.0,0.0,139.0,0.0,-661.0,0.0,800.0 base-pv-generators.xml,6.8,91.76,36000.0,24000.0,0.0,32239.0,8709.0,7508.0,0.0,575.0,6918.0,0.0,0.0,1738.0,2171.0,4620.0,0.0,0.0,20039.0,6112.0,7037.0,0.0,207.0,448.0,0.0,0.0,0.0,2293.0,622.0,0.0,3320.0,0.0,0.0,139.0,0.0,-661.0,0.0,800.0 base-pv.xml,6.8,91.76,36000.0,24000.0,0.0,32239.0,8709.0,7508.0,0.0,575.0,6918.0,0.0,0.0,1738.0,2171.0,4620.0,0.0,0.0,20039.0,6112.0,7037.0,0.0,207.0,448.0,0.0,0.0,0.0,2293.0,622.0,0.0,3320.0,0.0,0.0,139.0,0.0,-661.0,0.0,800.0 -base-residents-0-runperiod-1-month.xml,6.8,91.76,36000.0,24000.0,0.0,32239.0,8709.0,7508.0,0.0,575.0,6918.0,0.0,0.0,1738.0,2171.0,4620.0,0.0,0.0,20039.0,6112.0,7037.0,0.0,207.0,448.0,0.0,0.0,0.0,2293.0,622.0,0.0,3320.0,0.0,0.0,139.0,0.0,-661.0,0.0,800.0 base-residents-0.xml,6.8,91.76,36000.0,24000.0,0.0,32239.0,8709.0,7508.0,0.0,575.0,6918.0,0.0,0.0,1738.0,2171.0,4620.0,0.0,0.0,20039.0,6112.0,7037.0,0.0,207.0,448.0,0.0,0.0,0.0,2293.0,622.0,0.0,3320.0,0.0,0.0,139.0,0.0,-661.0,0.0,800.0 base-residents-1-misc-loads-large-uncommon.xml,6.8,91.76,36000.0,24000.0,0.0,33431.0,8742.0,7508.0,0.0,575.0,6918.0,0.0,0.0,1738.0,2171.0,5779.0,0.0,0.0,21277.0,6150.0,7037.0,0.0,207.0,448.0,0.0,0.0,0.0,2293.0,622.0,0.0,4520.0,0.0,0.0,139.0,0.0,-661.0,0.0,800.0 base-residents-1-misc-loads-large-uncommon2.xml,6.8,91.76,36000.0,24000.0,0.0,33431.0,8742.0,7508.0,0.0,575.0,6918.0,0.0,0.0,1738.0,2171.0,5779.0,0.0,0.0,21277.0,6150.0,7037.0,0.0,207.0,448.0,0.0,0.0,0.0,2293.0,622.0,0.0,4520.0,0.0,0.0,139.0,0.0,-661.0,0.0,800.0 diff --git a/workflow/tests/base_results/results_simulations_loads.csv b/workflow/tests/base_results/results_simulations_loads.csv index fe9e046176..7e033826de 100644 --- a/workflow/tests/base_results/results_simulations_loads.csv +++ b/workflow/tests/base_results/results_simulations_loads.csv @@ -411,7 +411,6 @@ base-pv-generators-battery-scheduled.xml,22.503,0.0,13.745,9.07,0.615,0.0,0.0,0. base-pv-generators-battery.xml,22.503,0.0,13.745,9.07,0.615,0.0,0.0,0.0,3.819,3.882,0.545,7.57,0.682,10.76,-13.571,0.0,0.0,0.0,8.363,-0.116,5.259,0.0,0.77,0.0,5.323,-8.475,-2.662,0.0,0.029,-0.188,-0.014,2.827,0.035,-0.632,10.837,0.0,0.0,0.0,-6.138,-0.112,-0.847,-3.884,-0.117,0.0,3.113,7.106,1.845 base-pv-generators.xml,22.503,0.0,13.745,9.07,0.615,0.0,0.0,0.0,3.819,3.882,0.545,7.57,0.682,10.76,-13.571,0.0,0.0,0.0,8.363,-0.116,5.259,0.0,0.77,0.0,5.323,-8.475,-2.662,0.0,0.029,-0.188,-0.014,2.827,0.035,-0.632,10.837,0.0,0.0,0.0,-6.138,-0.112,-0.847,-3.884,-0.117,0.0,3.113,7.106,1.845 base-pv.xml,22.503,0.0,13.745,9.07,0.615,0.0,0.0,0.0,3.819,3.882,0.545,7.57,0.682,10.76,-13.571,0.0,0.0,0.0,8.363,-0.116,5.259,0.0,0.77,0.0,5.323,-8.475,-2.662,0.0,0.029,-0.188,-0.014,2.827,0.035,-0.632,10.837,0.0,0.0,0.0,-6.138,-0.112,-0.847,-3.884,-0.117,0.0,3.113,7.106,1.845 -base-residents-0-runperiod-1-month.xml,7.9107,0.0,0.0,0.0,0.0511,0.0,0.0,0.0,0.5914,0.643,0.0909,1.746,0.1095,1.7781,-1.988,0.0,0.0,0.0,2.2357,-0.0004,1.0461,0.0,0.0,0.0,1.8131,-0.1971,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0 base-residents-0.xml,32.415,0.0,10.561,0.0,0.62,0.0,0.0,0.0,3.761,3.912,0.547,7.099,0.683,10.947,-14.892,0.0,0.0,0.0,8.358,-0.174,6.026,0.0,0.0,0.0,7.39,-1.573,0.0,0.0,0.311,0.056,0.019,3.03,0.087,0.17,9.539,0.0,0.0,0.0,-5.202,-0.169,-0.644,0.0,0.0,0.0,2.393,1.102,0.0 base-residents-1-misc-loads-large-uncommon.xml,21.522,0.0,14.588,3.416,0.614,0.0,0.0,0.0,3.822,3.877,0.545,7.617,0.681,10.741,-13.462,0.0,0.0,0.0,8.416,-0.111,5.233,0.0,0.766,0.0,5.114,-9.432,-2.643,0.0,-0.024,-0.232,-0.02,2.761,0.024,-0.764,10.946,0.0,0.0,0.0,-6.251,-0.107,-0.887,-4.03,-0.122,0.0,3.256,8.291,1.864 base-residents-1-misc-loads-large-uncommon2.xml,21.522,0.0,14.588,3.416,0.614,0.0,0.0,0.0,3.822,3.877,0.545,7.617,0.681,10.741,-13.462,0.0,0.0,0.0,8.416,-0.111,5.233,0.0,0.766,0.0,5.114,-9.432,-2.643,0.0,-0.024,-0.232,-0.02,2.761,0.024,-0.764,10.946,0.0,0.0,0.0,-6.251,-0.107,-0.887,-4.03,-0.122,0.0,3.256,8.291,1.864 diff --git a/workflow/tests/base_results/results_simulations_misc.csv b/workflow/tests/base_results/results_simulations_misc.csv index a942301e78..7babdf768c 100644 --- a/workflow/tests/base_results/results_simulations_misc.csv +++ b/workflow/tests/base_results/results_simulations_misc.csv @@ -411,7 +411,6 @@ base-pv-generators-battery-scheduled.xml,0.0,0.0,1354.7,998.0,11171.6,2563.5,208 base-pv-generators-battery.xml,0.0,0.0,1354.7,998.0,11171.6,2563.5,2125.8,3715.9,3715.9,23.71,18.744,80.224 base-pv-generators.xml,0.0,0.0,1354.7,998.0,11171.6,2563.5,2082.5,3612.6,3612.6,23.71,18.744,0.0 base-pv.xml,0.0,0.0,1354.7,998.0,11171.6,2563.5,2082.5,3612.6,3612.6,23.71,18.744,0.0 -base-residents-0-runperiod-1-month.xml,0.0,0.0,0.0,0.0,0.0,0.0,627.96,0.0,627.96,27.1458,0.0,0.0 base-residents-0.xml,0.0,0.0,0.0,0.0,0.0,0.0,609.5,1840.9,1840.9,25.712,14.806,0.0 base-residents-1-misc-loads-large-uncommon.xml,0.0,0.0,821.3,625.4,3446.7,1167.8,2361.4,4228.7,4228.7,23.649,19.133,0.0 base-residents-1-misc-loads-large-uncommon2.xml,0.0,0.0,821.3,625.4,3446.7,1167.8,2242.3,4045.5,4045.5,23.649,19.133,0.0