openstudio-common-measures 0.3.0 → 0.3.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (64) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +16 -0
  3. data/lib/openstudio/common_measures/version.rb +1 -1
  4. metadata +3 -63
  5. data/lib/measures/AddDaylightSensors/LICENSE.md +0 -27
  6. data/lib/measures/AddDaylightSensors/README.md +0 -136
  7. data/lib/measures/AddDaylightSensors/README.md.erb +0 -42
  8. data/lib/measures/AddDaylightSensors/measure.rb +0 -521
  9. data/lib/measures/AddDaylightSensors/measure.xml +0 -233
  10. data/lib/measures/EnableDemandControlledVentilation/LICENSE.md +0 -27
  11. data/lib/measures/EnableDemandControlledVentilation/README.md +0 -32
  12. data/lib/measures/EnableDemandControlledVentilation/README.md.erb +0 -42
  13. data/lib/measures/EnableDemandControlledVentilation/measure.rb +0 -154
  14. data/lib/measures/EnableDemandControlledVentilation/measure.xml +0 -99
  15. data/lib/measures/EnableEconomizerControl/LICENSE.md +0 -27
  16. data/lib/measures/EnableEconomizerControl/README.md +0 -48
  17. data/lib/measures/EnableEconomizerControl/README.md.erb +0 -42
  18. data/lib/measures/EnableEconomizerControl/measure.rb +0 -172
  19. data/lib/measures/EnableEconomizerControl/measure.xml +0 -124
  20. data/lib/measures/IncreaseInsulationRValueForExteriorWalls/LICENSE.md +0 -27
  21. data/lib/measures/IncreaseInsulationRValueForExteriorWalls/README.md +0 -64
  22. data/lib/measures/IncreaseInsulationRValueForExteriorWalls/README.md.erb +0 -42
  23. data/lib/measures/IncreaseInsulationRValueForExteriorWalls/measure.rb +0 -422
  24. data/lib/measures/IncreaseInsulationRValueForExteriorWalls/measure.xml +0 -150
  25. data/lib/measures/IncreaseInsulationRValueForExteriorWallsByPercentage/LICENSE.md +0 -27
  26. data/lib/measures/IncreaseInsulationRValueForExteriorWallsByPercentage/README.md +0 -32
  27. data/lib/measures/IncreaseInsulationRValueForExteriorWallsByPercentage/README.md.erb +0 -42
  28. data/lib/measures/IncreaseInsulationRValueForExteriorWallsByPercentage/measure.rb +0 -328
  29. data/lib/measures/IncreaseInsulationRValueForExteriorWallsByPercentage/measure.xml +0 -98
  30. data/lib/measures/IncreaseInsulationRValueForRoofs/LICENSE.md +0 -27
  31. data/lib/measures/IncreaseInsulationRValueForRoofs/README.md +0 -64
  32. data/lib/measures/IncreaseInsulationRValueForRoofs/README.md.erb +0 -42
  33. data/lib/measures/IncreaseInsulationRValueForRoofs/measure.rb +0 -422
  34. data/lib/measures/IncreaseInsulationRValueForRoofs/measure.xml +0 -143
  35. data/lib/measures/IncreaseInsulationRValueForRoofsByPercentage/LICENSE.md +0 -27
  36. data/lib/measures/IncreaseInsulationRValueForRoofsByPercentage/README.md +0 -32
  37. data/lib/measures/IncreaseInsulationRValueForRoofsByPercentage/README.md.erb +0 -42
  38. data/lib/measures/IncreaseInsulationRValueForRoofsByPercentage/measure.rb +0 -328
  39. data/lib/measures/IncreaseInsulationRValueForRoofsByPercentage/measure.xml +0 -98
  40. data/lib/measures/ReduceElectricEquipmentLoadsByPercentage/LICENSE.md +0 -27
  41. data/lib/measures/ReduceElectricEquipmentLoadsByPercentage/README.md +0 -97
  42. data/lib/measures/ReduceElectricEquipmentLoadsByPercentage/README.md.erb +0 -42
  43. data/lib/measures/ReduceElectricEquipmentLoadsByPercentage/measure.rb +0 -450
  44. data/lib/measures/ReduceElectricEquipmentLoadsByPercentage/measure.xml +0 -186
  45. data/lib/measures/ReduceLightingLoadsByPercentage/LICENSE.md +0 -27
  46. data/lib/measures/ReduceLightingLoadsByPercentage/README.md +0 -96
  47. data/lib/measures/ReduceLightingLoadsByPercentage/README.md.erb +0 -42
  48. data/lib/measures/ReduceLightingLoadsByPercentage/measure.rb +0 -513
  49. data/lib/measures/ReduceLightingLoadsByPercentage/measure.xml +0 -191
  50. data/lib/measures/ReduceSpaceInfiltrationByPercentage/LICENSE.md +0 -27
  51. data/lib/measures/ReduceSpaceInfiltrationByPercentage/README.md +0 -104
  52. data/lib/measures/ReduceSpaceInfiltrationByPercentage/README.md.erb +0 -42
  53. data/lib/measures/ReduceSpaceInfiltrationByPercentage/measure.rb +0 -349
  54. data/lib/measures/ReduceSpaceInfiltrationByPercentage/measure.xml +0 -181
  55. data/lib/measures/ReduceVentilationByPercentage/LICENSE.md +0 -27
  56. data/lib/measures/ReduceVentilationByPercentage/README.md +0 -40
  57. data/lib/measures/ReduceVentilationByPercentage/README.md.erb +0 -42
  58. data/lib/measures/ReduceVentilationByPercentage/measure.rb +0 -291
  59. data/lib/measures/ReduceVentilationByPercentage/measure.xml +0 -96
  60. data/lib/measures/create_variable_speed_rtu/LICENSE.md +0 -27
  61. data/lib/measures/create_variable_speed_rtu/README.md +0 -120
  62. data/lib/measures/create_variable_speed_rtu/README.md.erb +0 -42
  63. data/lib/measures/create_variable_speed_rtu/measure.rb +0 -539
  64. data/lib/measures/create_variable_speed_rtu/measure.xml +0 -207
@@ -1,98 +0,0 @@
1
- <measure>
2
- <schema_version>3.0</schema_version>
3
- <name>increase_insulation_r_value_for_exterior_walls_by_percentage</name>
4
- <uid>7508c563-22bd-49f4-b646-aac69b02a82d</uid>
5
- <version_id>9ef20b5d-2bdd-417b-a5d9-b7f77d509133</version_id>
6
- <version_modified>20200229T024318Z</version_modified>
7
- <xml_checksum>EB1A0C08</xml_checksum>
8
- <class_name>IncreaseInsulationRValueForExteriorWallsByPercentage</class_name>
9
- <display_name>Increase R-value of Insulation for Exterior Walls By a Specified Percentage</display_name>
10
- <description>Increase the R value of the insulation layer in wall constructinons by a percentage.</description>
11
- <modeler_description>Insulation later is inferred by finding the existing layer with the highest thermal resistance.</modeler_description>
12
- <arguments>
13
- <argument>
14
- <name>r_value</name>
15
- <display_name>Percentage Increase of R-value for Exterior Wall Insulation.</display_name>
16
- <type>Double</type>
17
- <required>true</required>
18
- <model_dependent>false</model_dependent>
19
- <default_value>30</default_value>
20
- </argument>
21
- </arguments>
22
- <outputs/>
23
- <provenances/>
24
- <tags>
25
- <tag>Envelope.Opaque</tag>
26
- </tags>
27
- <attributes>
28
- <attribute>
29
- <name>Measure Function</name>
30
- <value>Measure</value>
31
- <datatype>string</datatype>
32
- </attribute>
33
- <attribute>
34
- <name>Requires EnergyPlus Results</name>
35
- <value>false</value>
36
- <datatype>boolean</datatype>
37
- </attribute>
38
- <attribute>
39
- <name>Measure Type</name>
40
- <value>ModelMeasure</value>
41
- <datatype>string</datatype>
42
- </attribute>
43
- <attribute>
44
- <name>Uses SketchUp API</name>
45
- <value>false</value>
46
- <datatype>boolean</datatype>
47
- </attribute>
48
- </attributes>
49
- <files>
50
- <file>
51
- <filename>EnvelopeAndLoadTestModel_01.osm</filename>
52
- <filetype>osm</filetype>
53
- <usage_type>test</usage_type>
54
- <checksum>10AA8866</checksum>
55
- </file>
56
- <file>
57
- <filename>ReverseTranslatedModel.osm</filename>
58
- <filetype>osm</filetype>
59
- <usage_type>test</usage_type>
60
- <checksum>0B94403D</checksum>
61
- </file>
62
- <file>
63
- <filename>README.md.erb</filename>
64
- <filetype>erb</filetype>
65
- <usage_type>readmeerb</usage_type>
66
- <checksum>703C9964</checksum>
67
- </file>
68
- <file>
69
- <filename>README.md</filename>
70
- <filetype>md</filetype>
71
- <usage_type>readme</usage_type>
72
- <checksum>D4B90CE5</checksum>
73
- </file>
74
- <file>
75
- <filename>LICENSE.md</filename>
76
- <filetype>md</filetype>
77
- <usage_type>license</usage_type>
78
- <checksum>E0468DD6</checksum>
79
- </file>
80
- <file>
81
- <version>
82
- <software_program>OpenStudio</software_program>
83
- <identifier>2.0.0</identifier>
84
- <min_compatible>2.0.0</min_compatible>
85
- </version>
86
- <filename>measure.rb</filename>
87
- <filetype>rb</filetype>
88
- <usage_type>script</usage_type>
89
- <checksum>478DC9B7</checksum>
90
- </file>
91
- <file>
92
- <filename>IncreaseInsulationRValueForExteriorWalls_Test.rb</filename>
93
- <filetype>rb</filetype>
94
- <usage_type>test</usage_type>
95
- <checksum>01E73CC1</checksum>
96
- </file>
97
- </files>
98
- </measure>
@@ -1,27 +0,0 @@
1
- OpenStudio(R), Copyright (c) 2008-2020, Alliance for Sustainable Energy, LLC. All rights reserved.
2
-
3
- Redistribution and use in source and binary forms, with or without modification, are permitted
4
- provided that the following conditions are met:
5
-
6
- (1) Redistributions of source code must retain the above copyright notice, this list of conditions
7
- and the following disclaimer.
8
-
9
- (2) Redistributions in binary form must reproduce the above copyright notice, this list of conditions
10
- and the following disclaimer in the documentation and/or other materials provided with the distribution.
11
-
12
- (3) Neither the name of the copyright holder nor the names of any contributors may be used to endorse
13
- or promote products derived from this software without specific prior written permission from the
14
- respective party.
15
-
16
- (4) Other than as required in clauses (1) and (2), distributions in any form of modifications or other
17
- derivative works may not use the "OpenStudio" trademark, "OS", "os", or any other confusingly similar
18
- designation without specific prior written permission from Alliance for Sustainable Energy, LLC.
19
-
20
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED
21
- WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
22
- PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER, THE UNITED STATES GOVERNMENT,
23
- OR ANY CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24
- DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
25
- OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
26
- STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27
- SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
@@ -1,64 +0,0 @@
1
-
2
-
3
- ###### (Automatically generated documentation)
4
-
5
- # Increase R-value of Insulation for Roofs to a Specific Value
6
-
7
- ## Description
8
-
9
-
10
- ## Modeler Description
11
-
12
-
13
- ## Measure Type
14
- ModelMeasure
15
-
16
- ## Taxonomy
17
-
18
-
19
- ## Arguments
20
-
21
-
22
- ### Insulation R-value (ft^2*h*R/Btu).
23
-
24
- **Name:** r_value,
25
- **Type:** Double,
26
- **Units:** ,
27
- **Required:** true,
28
- **Model Dependent:** false
29
-
30
- ### Allow both increase and decrease in R-value to reach requested target?
31
-
32
- **Name:** allow_reduction,
33
- **Type:** Boolean,
34
- **Units:** ,
35
- **Required:** true,
36
- **Model Dependent:** false
37
-
38
- ### Increase in Material and Installation Costs for Construction per Area Used ($/ft^2).
39
-
40
- **Name:** material_cost_increase_ip,
41
- **Type:** Double,
42
- **Units:** ,
43
- **Required:** true,
44
- **Model Dependent:** false
45
-
46
- ### One Time Retrofit Cost to Add Insulation to Construction ($/ft^2).
47
-
48
- **Name:** one_time_retrofit_cost_ip,
49
- **Type:** Double,
50
- **Units:** ,
51
- **Required:** true,
52
- **Model Dependent:** false
53
-
54
- ### Year to Incur One Time Retrofit Cost (whole years).
55
-
56
- **Name:** years_until_retrofit_cost,
57
- **Type:** Integer,
58
- **Units:** ,
59
- **Required:** true,
60
- **Model Dependent:** false
61
-
62
-
63
-
64
-
@@ -1,42 +0,0 @@
1
- <%#= README.md.erb is used to auto-generate README.md. %>
2
- <%#= To manually maintain README.md throw away README.md.erb and manually edit README.md %>
3
- ###### (Automatically generated documentation)
4
-
5
- # <%= name %>
6
-
7
- ## Description
8
- <%= description %>
9
-
10
- ## Modeler Description
11
- <%= modelerDescription %>
12
-
13
- ## Measure Type
14
- <%= measureType %>
15
-
16
- ## Taxonomy
17
- <%= taxonomy %>
18
-
19
- ## Arguments
20
-
21
- <% arguments.each do |argument| %>
22
- ### <%= argument[:display_name] %>
23
- <%= argument[:description] %>
24
- **Name:** <%= argument[:name] %>,
25
- **Type:** <%= argument[:type] %>,
26
- **Units:** <%= argument[:units] %>,
27
- **Required:** <%= argument[:required] %>,
28
- **Model Dependent:** <%= argument[:model_dependent] %>
29
- <% end %>
30
-
31
- <% if arguments.size == 0 %>
32
- <%= "This measure does not have any user arguments" %>
33
- <% end %>
34
-
35
- <% if outputs.size > 0 %>
36
- ## Outputs
37
- <% output_names = [] %>
38
- <% outputs.each do |output| %>
39
- <% output_names << output[:display_name] %>
40
- <% end %>
41
- <%= output_names.join(", ") %>
42
- <% end %>
@@ -1,422 +0,0 @@
1
- # *******************************************************************************
2
- # OpenStudio(R), Copyright (c) 2008-2020, Alliance for Sustainable Energy, LLC.
3
- # All rights reserved.
4
- # Redistribution and use in source and binary forms, with or without
5
- # modification, are permitted provided that the following conditions are met:
6
- #
7
- # (1) Redistributions of source code must retain the above copyright notice,
8
- # this list of conditions and the following disclaimer.
9
- #
10
- # (2) Redistributions in binary form must reproduce the above copyright notice,
11
- # this list of conditions and the following disclaimer in the documentation
12
- # and/or other materials provided with the distribution.
13
- #
14
- # (3) Neither the name of the copyright holder nor the names of any contributors
15
- # may be used to endorse or promote products derived from this software without
16
- # specific prior written permission from the respective party.
17
- #
18
- # (4) Other than as required in clauses (1) and (2), distributions in any form
19
- # of modifications or other derivative works may not use the "OpenStudio"
20
- # trademark, "OS", "os", or any other confusingly similar designation without
21
- # specific prior written permission from Alliance for Sustainable Energy, LLC.
22
- #
23
- # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS
24
- # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
25
- # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26
- # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE
27
- # UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF
28
- # THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
29
- # EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
30
- # OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
31
- # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
32
- # STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
33
- # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
34
- # *******************************************************************************
35
-
36
- # start the measure
37
- class IncreaseInsulationRValueForRoofs < OpenStudio::Measure::ModelMeasure
38
- # define the name that a user will see
39
- def name
40
- return 'Increase R-value of Insulation for Roofs to a Specific Value'
41
- end
42
-
43
- # define the arguments that the user will input
44
- def arguments(model)
45
- args = OpenStudio::Measure::OSArgumentVector.new
46
-
47
- # make an argument insulation R-value
48
- r_value = OpenStudio::Measure::OSArgument.makeDoubleArgument('r_value', true)
49
- r_value.setDisplayName('Insulation R-value (ft^2*h*R/Btu).')
50
- r_value.setDefaultValue(30.0)
51
- args << r_value
52
-
53
- # make bool argument to allow both increase and decrease in R value
54
- allow_reduction = OpenStudio::Measure::OSArgument.makeBoolArgument('allow_reduction', true)
55
- allow_reduction.setDisplayName('Allow both increase and decrease in R-value to reach requested target?')
56
- allow_reduction.setDefaultValue(false)
57
- args << allow_reduction
58
-
59
- # make an argument for material and installation cost
60
- material_cost_increase_ip = OpenStudio::Measure::OSArgument.makeDoubleArgument('material_cost_increase_ip', true)
61
- material_cost_increase_ip.setDisplayName('Increase in Material and Installation Costs for Construction per Area Used ($/ft^2).')
62
- material_cost_increase_ip.setDefaultValue(0.0)
63
- args << material_cost_increase_ip
64
-
65
- # make an argument for demolition cost
66
- one_time_retrofit_cost_ip = OpenStudio::Measure::OSArgument.makeDoubleArgument('one_time_retrofit_cost_ip', true)
67
- one_time_retrofit_cost_ip.setDisplayName('One Time Retrofit Cost to Add Insulation to Construction ($/ft^2).')
68
- one_time_retrofit_cost_ip.setDefaultValue(0.0)
69
- args << one_time_retrofit_cost_ip
70
-
71
- # make an argument for duration in years until costs start
72
- years_until_retrofit_cost = OpenStudio::Measure::OSArgument.makeIntegerArgument('years_until_retrofit_cost', true)
73
- years_until_retrofit_cost.setDisplayName('Year to Incur One Time Retrofit Cost (whole years).')
74
- years_until_retrofit_cost.setDefaultValue(0)
75
- args << years_until_retrofit_cost
76
-
77
- return args
78
- end
79
-
80
- # define what happens when the measure is run
81
- def run(model, runner, user_arguments)
82
- super(model, runner, user_arguments)
83
-
84
- # use the built-in error checking
85
- if !runner.validateUserArguments(arguments(model), user_arguments)
86
- return false
87
- end
88
-
89
- # assign the user inputs to variables
90
- r_value = runner.getDoubleArgumentValue('r_value', user_arguments)
91
- allow_reduction = runner.getBoolArgumentValue('allow_reduction', user_arguments)
92
- material_cost_increase_ip = runner.getDoubleArgumentValue('material_cost_increase_ip', user_arguments)
93
- one_time_retrofit_cost_ip = runner.getDoubleArgumentValue('one_time_retrofit_cost_ip', user_arguments)
94
- years_until_retrofit_cost = runner.getIntegerArgumentValue('years_until_retrofit_cost', user_arguments)
95
-
96
- # set limit for minimum insulation. This is used to limit input and for inferring insulation layer in construction.
97
- min_expected_r_value_ip = 1 # ip units
98
-
99
- # check the R-value for reasonableness
100
- if (r_value < 0) || (r_value > 500)
101
- runner.registerError("The requested roof insulation R-value of #{r_value} ft^2*h*R/Btu was above the measure limit.")
102
- return false
103
- elsif r_value > 60
104
- runner.registerWarning("The requested roof insulation R-value of #{r_value} ft^2*h*R/Btu is abnormally high.")
105
- elsif r_value < min_expected_r_value_ip
106
- runner.registerWarning("The requested roof insulation R-value of #{r_value} ft^2*h*R/Btu is abnormally low.")
107
- end
108
-
109
- # check lifecycle arguments for reasonableness
110
- if (years_until_retrofit_cost < 0) && (years_until_retrofit_cost > 100)
111
- runner.registerError('Year to incur one time retrofit cost should be a non-negative integer less than or equal to 100.')
112
- end
113
-
114
- # short def to make numbers pretty (converts 4125001.25641 to 4,125,001.26 or 4,125,001). The definition be called through this measure
115
- def neat_numbers(number, roundto = 2) # round to 0 or 2)
116
- if roundto == 2
117
- number = format '%.2f', number
118
- else
119
- number = number.round
120
- end
121
- # regex to add commas
122
- number.to_s.reverse.gsub(/([0-9]{3}(?=([0-9])))/, '\\1,').reverse
123
- end
124
-
125
- # helper to make it easier to do unit conversions on the fly
126
- def unit_helper(number, from_unit_string, to_unit_string)
127
- converted_number = OpenStudio.convert(OpenStudio::Quantity.new(number, OpenStudio.createUnit(from_unit_string).get), OpenStudio.createUnit(to_unit_string).get).get.value
128
- end
129
-
130
- # convert r_value and material_cost to si for future use
131
- r_value_si = unit_helper(r_value, 'ft^2*h*R/Btu', 'm^2*K/W')
132
- material_cost_increase_si = unit_helper(material_cost_increase_ip, '1/ft^2', '1/m^2')
133
-
134
- # create an array of roofs and find range of starting construction R-value (not just insulation layer)
135
- surfaces = model.getSurfaces
136
- exterior_surfaces = []
137
- exterior_surface_constructions = []
138
- exterior_surface_construction_names = []
139
- roof_resistance = []
140
- surfaces.each do |surface|
141
- if (surface.outsideBoundaryCondition == 'Outdoors') && (surface.surfaceType == 'RoofCeiling')
142
- exterior_surfaces << surface
143
- roof_const = surface.construction.get
144
- # only add construction if it hasn't been added yet
145
- if !exterior_surface_construction_names.include?(roof_const.name.to_s)
146
- exterior_surface_constructions << roof_const.to_Construction.get
147
- end
148
- exterior_surface_construction_names << roof_const.name.to_s
149
- roof_resistance << 1 / roof_const.thermalConductance.to_f
150
- end
151
- end
152
-
153
- # nothing will be done if there are no exterior surfaces
154
- if exterior_surfaces.empty?
155
- runner.registerAsNotApplicable('Model does not have any roofs.')
156
- return true
157
- end
158
-
159
- # report strings for initial condition
160
- initial_string = []
161
- exterior_surface_constructions.uniq.each do |exterior_surface_construction|
162
- # unit conversion of roof insulation from SI units (M^2*K/W) to IP units (ft^2*h*R/Btu)
163
- initial_conductance_ip = unit_helper(1 / exterior_surface_construction.thermalConductance.to_f, 'm^2*K/W', 'ft^2*h*R/Btu')
164
- initial_string << "#{exterior_surface_construction.name} (R-#{(format '%.1f', initial_conductance_ip)})"
165
- end
166
- runner.registerInitialCondition("The building had #{initial_string.size} roof constructions: #{initial_string.sort.join(', ')}.")
167
-
168
- # hashes to track constructions and materials made by the measure, to avoid duplicates
169
- constructions_hash_old_new = {}
170
- constructions_hash_new_old = {} # used to get netArea of new construction and then cost objects of construction it replaced
171
- materials_hash = {}
172
-
173
- # array and counter for new constructions that are made, used for reporting final condition
174
- final_constructions_array = []
175
-
176
- # loop through all constructions and materials used on roofs, edit and clone
177
- exterior_surface_constructions.each do |exterior_surface_construction|
178
- construction_layers = exterior_surface_construction.layers
179
- max_thermal_resistance_material = ''
180
- max_thermal_resistance_material_index = ''
181
- materials_in_construction = construction_layers.map.with_index do |layer, i|
182
- { 'name' => layer.name.to_s,
183
- 'index' => i,
184
- 'nomass' => !layer.to_MasslessOpaqueMaterial.empty?,
185
- 'r_value' => layer.to_OpaqueMaterial.get.thermalResistance,
186
- 'mat' => layer }
187
- end
188
-
189
- no_mass_materials = materials_in_construction.select { |mat| mat['nomass'] == true }
190
- # measure will select the no mass material with the highest r-value as the insulation layer
191
- # if no mass materials are present, the measure will select the material with the highest r-value per inch
192
- if !no_mass_materials.empty?
193
- thermal_resistance_values = no_mass_materials.map { |mat| mat['r_value'] }
194
- max_mat_hash = no_mass_materials.select { |mat| mat['r_value'] >= thermal_resistance_values.max }
195
- else
196
- thermal_resistance_per_thickness_values = materials_in_construction.map { |mat| mat['r_value'] / mat['mat'].thickness }
197
- target_index = thermal_resistance_per_thickness_values.index(thermal_resistance_per_thickness_values.max)
198
- max_mat_hash = materials_in_construction.select { |mat| mat['index'] == target_index }
199
- thermal_resistance_values = materials_in_construction.map { |mat| mat['r_value'] }
200
- end
201
- max_thermal_resistance_material = max_mat_hash[0]['mat']
202
- max_thermal_resistance_material_index = max_mat_hash[0]['index']
203
- max_thermal_resistance = max_thermal_resistance_material.to_OpaqueMaterial.get.thermalResistance
204
-
205
- if max_thermal_resistance <= unit_helper(min_expected_r_value_ip, 'ft^2*h*R/Btu', 'm^2*K/W')
206
- runner.registerWarning("Construction '#{exterior_surface_construction.name}' does not appear to have an insulation layer and was not altered.")
207
- elsif (max_thermal_resistance >= r_value_si) && !allow_reduction
208
- runner.registerInfo("The insulation layer of construction #{exterior_surface_construction.name} exceeds the requested R-Value. It was not altered.")
209
- else
210
- # clone the construction
211
- final_construction = exterior_surface_construction.clone(model)
212
- final_construction = final_construction.to_Construction.get
213
- final_construction.setName("#{exterior_surface_construction.name} adj roof insulation")
214
- final_constructions_array << final_construction
215
-
216
- # loop through lifecycle costs getting total costs under "Construction" or "Salvage" category and add to counter if occurs during year 0
217
- const_LCCs = final_construction.lifeCycleCosts
218
- cost_added = false
219
- const_LCC_cat_const = false
220
- updated_cost_si = 0
221
- const_LCCs.each do |const_LCC|
222
- if (const_LCC.category == 'Construction') && (material_cost_increase_si != 0)
223
- const_LCC_cat_const = true # need this test to add proper lcc if it didn't exist to start with
224
- # if multiple LCC objects associated with construction only adjust the cost of one of them.
225
- if !cost_added
226
- const_LCC.setCost(const_LCC.cost + material_cost_increase_si)
227
- else
228
- runner.registerInfo("More than one LifeCycleCost object with a category of Construction was associated with #{final_construction.name}. Cost was only adjusted for one of the LifeCycleCost objects.")
229
- end
230
- updated_cost_si += const_LCC.cost
231
- end
232
- end
233
-
234
- if cost_added
235
- runner.registerInfo("Adjusting material and installation cost for #{final_construction.name} to #{neat_numbers(unit_helper(updated_cost_si, '1/m^2', '1/ft^2'))} ($/ft^2).")
236
- end
237
-
238
- # add construction object if it didnt exist to start with and a cost increase was requested
239
- if (const_LCC_cat_const == false) && (material_cost_increase_si != 0)
240
- lcc_for_uncosted_const = OpenStudio::Model::LifeCycleCost.createLifeCycleCost('LCC_increase_insulation', final_construction, material_cost_increase_si, 'CostPerArea', 'Construction', 20, 0).get
241
- runner.registerInfo("No material or installation costs existed for #{final_construction.name}. Created a new LifeCycleCost object with a material and installation cost of #{neat_numbers(unit_helper(lcc_for_uncosted_const.cost, '1/m^2', '1/ft^2'))} ($/ft^2). Assumed capitol cost in first year, an expected life of 20 years, and no O & M costs.")
242
- end
243
-
244
- # add one time cost if requested
245
- if one_time_retrofit_cost_ip > 0
246
- one_time_retrofit_cost_si = unit_helper(one_time_retrofit_cost_ip, '1/ft^2', '1/m^2')
247
- lcc_retrofit_specific = OpenStudio::Model::LifeCycleCost.createLifeCycleCost('LCC_retrofit_specific', final_construction, one_time_retrofit_cost_si, 'CostPerArea', 'Construction', 0, years_until_retrofit_cost).get # using 0 for repeat period since one time cost.
248
- runner.registerInfo("Adding one time cost of #{neat_numbers(unit_helper(lcc_retrofit_specific.cost, '1/m^2', '1/ft^2'))} ($/ft^2) related to retrofit of roof insulation.")
249
- end
250
-
251
- # push to hashes
252
- constructions_hash_old_new[exterior_surface_construction.name.to_s] = final_construction
253
- constructions_hash_new_old[final_construction] = exterior_surface_construction # push the object to hash key vs. name
254
-
255
- # find already cloned insulation material and link to construction
256
- target_material = max_thermal_resistance_material
257
- found_material = false
258
- materials_hash.each do |orig, new|
259
- if target_material.name.to_s == orig
260
- new_material = new
261
- materials_hash[max_thermal_resistance_material.name.to_s] = new_material
262
- final_construction.eraseLayer(max_thermal_resistance_material_index)
263
- final_construction.insertLayer(max_thermal_resistance_material_index, new_material)
264
- found_material = true
265
- end
266
- end
267
-
268
- # clone and edit insulation material and link to construction
269
- if found_material == false
270
- new_material = max_thermal_resistance_material.clone(model)
271
- new_material = new_material.to_OpaqueMaterial.get
272
- new_material.setName("#{max_thermal_resistance_material.name}_R-value #{r_value} (ft^2*h*R/Btu)")
273
- materials_hash[max_thermal_resistance_material.name.to_s] = new_material
274
- final_construction.eraseLayer(max_thermal_resistance_material_index)
275
- final_construction.insertLayer(max_thermal_resistance_material_index, new_material)
276
- runner.registerInfo("For construction'#{final_construction.name}', material'#{new_material.name}' was altered.")
277
-
278
- # edit insulation material
279
- new_material_matt = new_material.to_Material
280
- if !new_material_matt.empty?
281
- starting_thickness = new_material_matt.get.thickness
282
- target_thickness = starting_thickness * r_value_si / thermal_resistance_values.max
283
- final_thickness = new_material_matt.get.setThickness(target_thickness)
284
- end
285
- new_material_massless = new_material.to_MasslessOpaqueMaterial
286
- if !new_material_massless.empty?
287
- final_thermal_resistance = new_material_massless.get.setThermalResistance(r_value_si)
288
- end
289
- new_material_airgap = new_material.to_AirGap
290
- if !new_material_airgap.empty?
291
- final_thermal_resistance = new_material_airgap.get.setThermalResistance(r_value_si)
292
- end
293
- end
294
- end
295
- end
296
-
297
- # loop through construction sets used in the model
298
- default_construction_sets = model.getDefaultConstructionSets
299
- default_construction_sets.each do |default_construction_set|
300
- if default_construction_set.directUseCount > 0
301
- default_surface_const_set = default_construction_set.defaultExteriorSurfaceConstructions
302
- if !default_surface_const_set.empty?
303
- starting_construction = default_surface_const_set.get.roofCeilingConstruction
304
-
305
- # creating new default construction set
306
- new_default_construction_set = default_construction_set.clone(model)
307
- new_default_construction_set = new_default_construction_set.to_DefaultConstructionSet.get
308
- new_default_construction_set.setName("#{default_construction_set.name} adj roof insulation")
309
-
310
- # create new surface set and link to construction set
311
- new_default_surface_const_set = default_surface_const_set.get.clone(model)
312
- new_default_surface_const_set = new_default_surface_const_set.to_DefaultSurfaceConstructions.get
313
- new_default_surface_const_set.setName("#{default_surface_const_set.get.name} adj roof insulation")
314
- new_default_construction_set.setDefaultExteriorSurfaceConstructions(new_default_surface_const_set)
315
-
316
- # use the hash to find the proper construction and link to new_default_surface_const_set
317
- target_const = new_default_surface_const_set.roofCeilingConstruction
318
- if !target_const.empty?
319
- target_const = target_const.get.name.to_s
320
- found_const_flag = false
321
- constructions_hash_old_new.each do |orig, new|
322
- if target_const == orig
323
- final_construction = new
324
- new_default_surface_const_set.setRoofCeilingConstruction(final_construction)
325
- found_const_flag = true
326
- end
327
- end
328
- if found_const_flag == false # this should never happen but is just an extra test in case something goes wrong with the measure code
329
- runner.registerWarning("Measure couldn't find the construction named '#{target_const}' in the exterior surface hash.")
330
- end
331
- end
332
-
333
- # swap all uses of the old construction set for the new
334
- construction_set_sources = default_construction_set.sources
335
- construction_set_sources.each do |construction_set_source|
336
- building_source = construction_set_source.to_Building
337
- # if statement for each type of object than can use a DefaultConstructionSet
338
- if !building_source.empty?
339
- building_source = building_source.get
340
- building_source.setDefaultConstructionSet(new_default_construction_set)
341
- end
342
- building_story_source = construction_set_source.to_BuildingStory
343
- if !building_story_source.empty?
344
- building_story_source = building_story_source.get
345
- building_story_source.setDefaultConstructionSet(new_default_construction_set)
346
- end
347
- space_type_source = construction_set_source.to_SpaceType
348
- if !space_type_source.empty?
349
- space_type_source = space_type_source.get
350
- space_type_source.setDefaultConstructionSet(new_default_construction_set)
351
- end
352
- space_source = construction_set_source.to_Space
353
- if !space_source.empty?
354
- space_source = space_source.get
355
- space_source.setDefaultConstructionSet(new_default_construction_set)
356
- end
357
- end
358
-
359
- end
360
- end
361
- end
362
-
363
- # link cloned and edited constructions for surfaces with hard assigned constructions
364
- exterior_surfaces.each do |exterior_surface|
365
- if !exterior_surface.isConstructionDefaulted && !exterior_surface.construction.empty?
366
-
367
- # use the hash to find the proper construction and link to surface
368
- target_const = exterior_surface.construction
369
- if !target_const.empty?
370
- target_const = target_const.get.name.to_s
371
- constructions_hash_old_new.each do |orig, new|
372
- if target_const == orig
373
- final_construction = new
374
- exterior_surface.setConstruction(final_construction)
375
- end
376
- end
377
- end
378
-
379
- end
380
- end
381
-
382
- # report strings for final condition
383
- final_string = [] # not all exterior roof constructions, but only new ones made. If roof didn't have insulation and was not altered we don't want to show it
384
- affected_area_si = 0
385
- totalCost_of_affected_area = 0
386
- yr0_capital_totalCosts = 0
387
- final_constructions_array.each do |final_construction|
388
- # unit conversion of roof insulation from SI units (M^2*K/W) to IP units (ft^2*h*R/Btu)
389
- final_conductance_ip = unit_helper(1 / final_construction.thermalConductance.to_f, 'm^2*K/W', 'ft^2*h*R/Btu')
390
- final_string << "#{final_construction.name} (R-#{(format '%.1f', final_conductance_ip)})"
391
- affected_area_si += final_construction.getNetArea
392
-
393
- # loop through lifecycle costs getting total costs under "Construction" or "Salvage" category and add to counter if occurs during year 0
394
- const_LCCs = final_construction.lifeCycleCosts
395
- const_LCCs.each do |const_LCC|
396
- if (const_LCC.category == 'Construction') || (const_LCC.category == 'Salvage')
397
- if const_LCC.yearsFromStart == 0
398
- yr0_capital_totalCosts += const_LCC.totalCost
399
- end
400
- end
401
- end
402
- end
403
-
404
- # add not applicable test if there were exterior roof constructions but non of them were altered (already enough insulation or doesn't look like insulated wall)
405
- if affected_area_si == 0
406
- runner.registerAsNotApplicable('No roofs were altered.')
407
- return true
408
- # affected_area_ip = affected_area_si
409
- else
410
- # ip construction area for reporting
411
- affected_area_ip = unit_helper(affected_area_si, 'm^2', 'ft^2')
412
- end
413
-
414
- # report final condition
415
- runner.registerFinalCondition("The existing insulation for roofs was changed to R-#{r_value}. This was accomplished for an initial cost of #{one_time_retrofit_cost_ip} ($/sf) and an increase of #{material_cost_increase_ip} ($/sf) for construction. This was applied to #{neat_numbers(affected_area_ip, 0)} (ft^2) across #{final_string.size} roof constructions: #{final_string.sort.join(', ')}.")
416
-
417
- return true
418
- end
419
- end
420
-
421
- # this allows the measure to be used by the application
422
- IncreaseInsulationRValueForRoofs.new.registerWithApplication