tbd 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.gitattributes +3 -0
- data/.github/workflows/pull_request.yml +72 -0
- data/.gitignore +23 -0
- data/.rspec +3 -0
- data/Gemfile +3 -0
- data/LICENSE.md +21 -0
- data/README.md +154 -0
- data/Rakefile +60 -0
- data/json/midrise.json +64 -0
- data/json/tbd_5ZoneNoHVAC.json +19 -0
- data/json/tbd_5ZoneNoHVAC_btap.json +91 -0
- data/json/tbd_seb_n2.json +41 -0
- data/json/tbd_seb_n4.json +57 -0
- data/json/tbd_warehouse10.json +24 -0
- data/json/tbd_warehouse5.json +37 -0
- data/lib/measures/tbd/LICENSE.md +21 -0
- data/lib/measures/tbd/README.md +136 -0
- data/lib/measures/tbd/README.md.erb +42 -0
- data/lib/measures/tbd/docs/.gitkeep +1 -0
- data/lib/measures/tbd/measure.rb +327 -0
- data/lib/measures/tbd/measure.xml +460 -0
- data/lib/measures/tbd/resources/geo.rb +714 -0
- data/lib/measures/tbd/resources/geometry.rb +351 -0
- data/lib/measures/tbd/resources/model.rb +1431 -0
- data/lib/measures/tbd/resources/oslog.rb +381 -0
- data/lib/measures/tbd/resources/psi.rb +2229 -0
- data/lib/measures/tbd/resources/tbd.rb +55 -0
- data/lib/measures/tbd/resources/transformation.rb +121 -0
- data/lib/measures/tbd/resources/ua.rb +986 -0
- data/lib/measures/tbd/resources/utils.rb +1636 -0
- data/lib/measures/tbd/resources/version.rb +3 -0
- data/lib/measures/tbd/tests/tbd_full_PSI.json +17 -0
- data/lib/measures/tbd/tests/tbd_tests.rb +222 -0
- data/lib/tbd/geo.rb +714 -0
- data/lib/tbd/psi.rb +2229 -0
- data/lib/tbd/ua.rb +986 -0
- data/lib/tbd/version.rb +25 -0
- data/lib/tbd.rb +93 -0
- data/sponsors/canada.png +0 -0
- data/sponsors/quebec.png +0 -0
- data/tbd.gemspec +43 -0
- data/tbd.schema.json +571 -0
- data/v291_MacOS.md +110 -0
- metadata +191 -0
@@ -0,0 +1,1636 @@
|
|
1
|
+
# BSD 3-Clause License
|
2
|
+
#
|
3
|
+
# Copyright (c) 2022, Denis Bourgeois
|
4
|
+
# All rights reserved.
|
5
|
+
#
|
6
|
+
# Redistribution and use in source and binary forms, with or without
|
7
|
+
# modification, are permitted provided that the following conditions are met:
|
8
|
+
#
|
9
|
+
# 1. Redistributions of source code must retain the above copyright notice, this
|
10
|
+
# list of conditions and the following disclaimer.
|
11
|
+
#
|
12
|
+
# 2. Redistributions in binary form must reproduce the above copyright notice,
|
13
|
+
# this list of conditions and the following disclaimer in the documentation
|
14
|
+
# and/or other materials provided with the distribution.
|
15
|
+
#
|
16
|
+
# 3. Neither the name of the copyright holder nor the names of its
|
17
|
+
# contributors may be used to endorse or promote products derived from
|
18
|
+
# this software without specific prior written permission.
|
19
|
+
#
|
20
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
21
|
+
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
22
|
+
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
23
|
+
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
24
|
+
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
25
|
+
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
26
|
+
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
27
|
+
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
28
|
+
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
29
|
+
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
30
|
+
|
31
|
+
require "openstudio"
|
32
|
+
|
33
|
+
module OSut
|
34
|
+
extend OSlg # DEBUG for devs; WARN/ERROR for users (bad OS input)
|
35
|
+
|
36
|
+
TOL = 0.01
|
37
|
+
TOL2 = TOL * TOL
|
38
|
+
DBG = OSut::DEBUG # mainly to flag invalid arguments to devs (buggy code)
|
39
|
+
INF = OSut::INFO # not currently used in OSut
|
40
|
+
WRN = OSut::WARN # WARN users of 'iffy' .osm inputs (yet not critical)
|
41
|
+
ERR = OSut::ERROR # flag invalid .osm inputs (then exit via 'return')
|
42
|
+
FTL = OSut::FATAL # not currently used in OSut
|
43
|
+
NS = "nameString" # OpenStudio IdfObject nameString method
|
44
|
+
|
45
|
+
# This first set of utilities (~750 lines) help distinguishing spaces that
|
46
|
+
# are directly vs indirectly CONDITIONED, vs SEMI-HEATED. The solution here
|
47
|
+
# relies as much as possible on space conditioning categories found in
|
48
|
+
# standards like ASHRAE 90.1 and energy codes like the Canadian NECB editions.
|
49
|
+
# Both documents share many similarities, regardless of nomenclature. There
|
50
|
+
# are however noticeable differences between approaches on how a space is
|
51
|
+
# tagged as falling into one of the aforementioned categories. First, an
|
52
|
+
# overview of 90.1 requirements, with some minor edits for brevity/emphasis:
|
53
|
+
#
|
54
|
+
# www.pnnl.gov/main/publications/external/technical_reports/PNNL-26917.pdf
|
55
|
+
#
|
56
|
+
# 3.2.1. General Information - SPACE CONDITIONING CATEGORY
|
57
|
+
#
|
58
|
+
# - CONDITIONED space: an ENCLOSED space that has a heating and/or
|
59
|
+
# cooling system of sufficient size to maintain temperatures suitable
|
60
|
+
# for HUMAN COMFORT:
|
61
|
+
# - COOLED: cooled by a system >= 10 W/m2
|
62
|
+
# - HEATED: heated by a system e.g., >= 50 W/m2 in Climate Zone CZ-7
|
63
|
+
# - INDIRECTLY: heated or cooled via adjacent space(s) provided:
|
64
|
+
# - UA of adjacent surfaces > UA of other surfaces
|
65
|
+
# or
|
66
|
+
# - intentional air transfer from HEATED/COOLED space > 3 ACH
|
67
|
+
#
|
68
|
+
# ... includes plenums, atria, etc.
|
69
|
+
#
|
70
|
+
# - SEMI-HEATED space: an ENCLOSED space that has a heating system
|
71
|
+
# >= 10 W/m2, yet NOT a CONDITIONED space (see above).
|
72
|
+
#
|
73
|
+
# - UNCONDITIONED space: an ENCLOSED space that is NOT a conditioned
|
74
|
+
# space or a SEMI-HEATED space (see above).
|
75
|
+
#
|
76
|
+
# NOTE: Crawlspaces, attics, and parking garages with natural or
|
77
|
+
# mechanical ventilation are considered UNENCLOSED spaces.
|
78
|
+
#
|
79
|
+
# 2.3.3 Modeling Requirements: surfaces adjacent to UNENCLOSED spaces
|
80
|
+
# shall be treated as exterior surfaces. All other UNENCLOSED surfaces
|
81
|
+
# are to be modeled as is in both proposed and baseline models. For
|
82
|
+
# instance, modeled fenestration in UNENCLOSED spaces would not be
|
83
|
+
# factored in WWR calculations.
|
84
|
+
#
|
85
|
+
#
|
86
|
+
# Related NECB definitions and concepts, starting with CONDITIONED space:
|
87
|
+
#
|
88
|
+
# "[...] the temperature of which is controlled to limit variation in
|
89
|
+
# response to the exterior ambient temperature by the provision, either
|
90
|
+
# DIRECTLY or INDIRECTLY, of heating or cooling [...]". Although criteria
|
91
|
+
# differ (e.g., not sizing-based), the general idea is sufficiently similar
|
92
|
+
# to ASHRAE 90.1 (e.g., heating and/or cooling based, no distinction for
|
93
|
+
# INDIRECTLY conditioned spaces like plenums).
|
94
|
+
#
|
95
|
+
# SEMI-HEATED spaces are also a defined NECB term, but again the distinction
|
96
|
+
# is based on desired/intended design space setpoint temperatures - not
|
97
|
+
# system sizing criteria. No further treatment is implemented here to
|
98
|
+
# distinguish SEMI-HEATED from CONDITIONED spaces.
|
99
|
+
#
|
100
|
+
# The single NECB criterion distinguishing UNCONDITIONED ENCLOSED spaces
|
101
|
+
# (such as vestibules) from UNENCLOSED spaces (such as attics) remains the
|
102
|
+
# intention to ventilate - or rather to what degree. Regardless, the methods
|
103
|
+
# here are designed to process both classifications in the same way, namely by
|
104
|
+
# focusing on adjacent surfaces to CONDITIONED (or SEMI-HEATED) spaces as part
|
105
|
+
# of the building envelope.
|
106
|
+
|
107
|
+
# In light of the above, methods here are designed without a priori knowledge
|
108
|
+
# of explicit system sizing choices or access to iterative autosizing
|
109
|
+
# processes. As discussed in greater detail elswhere, methods are developed to
|
110
|
+
# rely on zoning info and/or "intended" temperature setpoints.
|
111
|
+
#
|
112
|
+
# For an OpenStudio model (OSM) in an incomplete or preliminary state, e.g.
|
113
|
+
# holding fully-formed ENCLOSED spaces without thermal zoning information or
|
114
|
+
# setpoint temperatures (early design stage assessments of form, porosity or
|
115
|
+
# envelope), all OSM spaces will be considered CONDITIONED, presuming
|
116
|
+
# setpoints of ~21°C (heating) and ~24°C (cooling).
|
117
|
+
#
|
118
|
+
# If ANY valid space/zone-specific temperature setpoints are found in the OSM,
|
119
|
+
# spaces/zones WITHOUT valid heating or cooling setpoints are considered as
|
120
|
+
# UNCONDITIONED or UNENCLOSED spaces (like attics), or INDIRECTLY CONDITIONED
|
121
|
+
# spaces (like plenums), see "plenum?" method.
|
122
|
+
|
123
|
+
##
|
124
|
+
# Return min & max values of a schedule (ruleset).
|
125
|
+
#
|
126
|
+
# @param sched [OpenStudio::Model::ScheduleRuleset] schedule
|
127
|
+
#
|
128
|
+
# @return [Hash] min: (Float), max: (Float)
|
129
|
+
# @return [Hash] min: nil, max: nil (if invalid input)
|
130
|
+
def scheduleRulesetMinMax(sched = nil)
|
131
|
+
# Largely inspired from David Goldwasser's
|
132
|
+
# "schedule_ruleset_annual_min_max_value":
|
133
|
+
#
|
134
|
+
# github.com/NREL/openstudio-standards/blob/
|
135
|
+
# 99cf713750661fe7d2082739f251269c2dfd9140/lib/openstudio-standards/
|
136
|
+
# standards/Standards.ScheduleRuleset.rb#L124
|
137
|
+
mth = "OSut::#{__callee__}"
|
138
|
+
cl = OpenStudio::Model::ScheduleRuleset
|
139
|
+
res = { min: nil, max: nil }
|
140
|
+
|
141
|
+
return invalid("sched", mth, 1, DBG, res) unless sched.respond_to?(NS)
|
142
|
+
id = sched.nameString
|
143
|
+
return mismatch(id, sched, cl, mth, DBG, res) unless sched.is_a?(cl)
|
144
|
+
|
145
|
+
profiles = []
|
146
|
+
profiles << sched.defaultDaySchedule
|
147
|
+
sched.scheduleRules.each { |rule| profiles << rule.daySchedule }
|
148
|
+
|
149
|
+
profiles.each do |profile|
|
150
|
+
id = profile.nameString
|
151
|
+
|
152
|
+
profile.values.each do |val|
|
153
|
+
ok = val.is_a?(Numeric)
|
154
|
+
log(WRN, "Skipping non-numeric value in '#{id}' (#{mth})") unless ok
|
155
|
+
next unless ok
|
156
|
+
|
157
|
+
res[:min] = val unless res[:min]
|
158
|
+
res[:min] = val if res[:min] > val
|
159
|
+
res[:max] = val unless res[:max]
|
160
|
+
res[:max] = val if res[:max] < val
|
161
|
+
end
|
162
|
+
end
|
163
|
+
|
164
|
+
valid = res[:min] && res[:max]
|
165
|
+
log(ERR, "Invalid MIN/MAX in '#{id}' (#{mth})") unless valid
|
166
|
+
|
167
|
+
res
|
168
|
+
end
|
169
|
+
|
170
|
+
##
|
171
|
+
# Return min & max values of a schedule (constant).
|
172
|
+
#
|
173
|
+
# @param sched [OpenStudio::Model::ScheduleConstant] schedule
|
174
|
+
#
|
175
|
+
# @return [Hash] min: (Float), max: (Float)
|
176
|
+
# @return [Hash] min: nil, max: nil (if invalid input)
|
177
|
+
def scheduleConstantMinMax(sched = nil)
|
178
|
+
# Largely inspired from David Goldwasser's
|
179
|
+
# "schedule_constant_annual_min_max_value":
|
180
|
+
#
|
181
|
+
# github.com/NREL/openstudio-standards/blob/
|
182
|
+
# 99cf713750661fe7d2082739f251269c2dfd9140/lib/openstudio-standards/
|
183
|
+
# standards/Standards.ScheduleConstant.rb#L21
|
184
|
+
mth = "OSut::#{__callee__}"
|
185
|
+
cl = OpenStudio::Model::ScheduleConstant
|
186
|
+
res = { min: nil, max: nil }
|
187
|
+
|
188
|
+
return invalid("sched", mth, 1, DBG, res) unless sched.respond_to?(NS)
|
189
|
+
id = sched.nameString
|
190
|
+
return mismatch(id, sched, cl, mth, DBG, res) unless sched.is_a?(cl)
|
191
|
+
|
192
|
+
valid = sched.value.is_a?(Numeric)
|
193
|
+
mismatch("'#{id}' value", sched.value, Numeric, mth, ERR, res) unless valid
|
194
|
+
res[:min] = sched.value
|
195
|
+
res[:max] = sched.value
|
196
|
+
|
197
|
+
res
|
198
|
+
end
|
199
|
+
|
200
|
+
##
|
201
|
+
# Return min & max values of a schedule (compact).
|
202
|
+
#
|
203
|
+
# @param sched [OpenStudio::Model::ScheduleCompact] schedule
|
204
|
+
#
|
205
|
+
# @return [Hash] min: (Float), max: (Float)
|
206
|
+
# @return [Hash] min: nil, max: nil (if invalid input)
|
207
|
+
def scheduleCompactMinMax(sched = nil)
|
208
|
+
# Largely inspired from Andrew Parker's
|
209
|
+
# "schedule_compact_annual_min_max_value":
|
210
|
+
#
|
211
|
+
# github.com/NREL/openstudio-standards/blob/
|
212
|
+
# 99cf713750661fe7d2082739f251269c2dfd9140/lib/openstudio-standards/
|
213
|
+
# standards/Standards.ScheduleCompact.rb#L8
|
214
|
+
mth = "OSut::#{__callee__}"
|
215
|
+
cl = OpenStudio::Model::ScheduleCompact
|
216
|
+
vals = []
|
217
|
+
prev_str = ""
|
218
|
+
res = { min: nil, max: nil }
|
219
|
+
|
220
|
+
return invalid("sched", mth, 1, DBG, res) unless sched.respond_to?(NS)
|
221
|
+
id = sched.nameString
|
222
|
+
return mismatch(id, sched, cl, mth, DBG, res) unless sched.is_a?(cl)
|
223
|
+
|
224
|
+
sched.extensibleGroups.each do |eg|
|
225
|
+
if prev_str.include?("until")
|
226
|
+
vals << eg.getDouble(0).get unless eg.getDouble(0).empty?
|
227
|
+
end
|
228
|
+
|
229
|
+
str = eg.getString(0)
|
230
|
+
prev_str = str.get.downcase unless str.empty?
|
231
|
+
end
|
232
|
+
|
233
|
+
return empty("'#{id}' values", mth, ERR, res) if vals.empty?
|
234
|
+
ok = vals.min.is_a?(Numeric) && vals.max.is_a?(Numeric)
|
235
|
+
log(ERR, "Non-numeric values in '#{id}' (#{mth})") unless ok
|
236
|
+
return res unless ok
|
237
|
+
res[:min] = vals.min
|
238
|
+
res[:max] = vals.max
|
239
|
+
|
240
|
+
res
|
241
|
+
end
|
242
|
+
|
243
|
+
##
|
244
|
+
# Return min & max values for schedule (interval).
|
245
|
+
#
|
246
|
+
# @param sched [OpenStudio::Model::ScheduleInterval] schedule
|
247
|
+
#
|
248
|
+
# @return [Hash] min: (Float), max: (Float)
|
249
|
+
# @return [Hash] min: nil, max: nil (if invalid input)
|
250
|
+
def scheduleIntervalMinMax(sched = nil)
|
251
|
+
mth = "OSut::#{__callee__}"
|
252
|
+
cl = OpenStudio::Model::ScheduleInterval
|
253
|
+
vals = []
|
254
|
+
prev_str = ""
|
255
|
+
res = { min: nil, max: nil }
|
256
|
+
|
257
|
+
return invalid("sched", mth, 1, DBG, res) unless sched.respond_to?(NS)
|
258
|
+
id = sched.nameString
|
259
|
+
return mismatch(id, sched, cl, mth, DBG, res) unless sched.is_a?(cl)
|
260
|
+
vals = sched.timeSeries.values
|
261
|
+
ok = vals.min.is_a?(Numeric) && vals.max.is_a?(Numeric)
|
262
|
+
log(ERR, "Non-numeric values in '#{id}' (#{mth})") unless ok
|
263
|
+
return res unless ok
|
264
|
+
res[:min] = vals.min
|
265
|
+
res[:max] = vals.max
|
266
|
+
|
267
|
+
res
|
268
|
+
end
|
269
|
+
|
270
|
+
##
|
271
|
+
# Return max zone heating temperature schedule setpoint [°C] and whether
|
272
|
+
# zone has active dual setpoint thermostat.
|
273
|
+
#
|
274
|
+
# @param zone [OpenStudio::Model::ThermalZone] a thermal zone
|
275
|
+
#
|
276
|
+
# @return [Hash] spt: (Float), dual: (Bool)
|
277
|
+
# @return [Hash] spt: nil, dual: false (if invalid input)
|
278
|
+
def maxHeatScheduledSetpoint(zone = nil)
|
279
|
+
# Largely inspired from Parker & Marrec's "thermal_zone_heated?" procedure.
|
280
|
+
# The solution here is a tad more relaxed to encompass SEMI-HEATED zones as
|
281
|
+
# per Canadian NECB criteria (basically any space with at least 10 W/m2 of
|
282
|
+
# installed heating equipement, i.e. below freezing in Canada).
|
283
|
+
#
|
284
|
+
# github.com/NREL/openstudio-standards/blob/
|
285
|
+
# 58964222d25783e9da4ae292e375fb0d5c902aa5/lib/openstudio-standards/
|
286
|
+
# standards/Standards.ThermalZone.rb#L910
|
287
|
+
mth = "OSut::#{__callee__}"
|
288
|
+
cl = OpenStudio::Model::ThermalZone
|
289
|
+
res = { spt: nil, dual: false }
|
290
|
+
|
291
|
+
return invalid("zone", mth, 1, DBG, res) unless zone.respond_to?(NS)
|
292
|
+
id = zone.nameString
|
293
|
+
return mismatch(id, zone, cl, mth, DBG, res) unless zone.is_a?(cl)
|
294
|
+
|
295
|
+
# Zone radiant heating? Get schedule from radiant system.
|
296
|
+
zone.equipment.each do |equip|
|
297
|
+
sched = nil
|
298
|
+
|
299
|
+
unless equip.to_ZoneHVACHighTemperatureRadiant.empty?
|
300
|
+
equip = equip.to_ZoneHVACHighTemperatureRadiant.get
|
301
|
+
|
302
|
+
unless equip.heatingSetpointTemperatureSchedule.empty?
|
303
|
+
sched = equip.heatingSetpointTemperatureSchedule.get
|
304
|
+
end
|
305
|
+
end
|
306
|
+
|
307
|
+
unless equip.to_ZoneHVACLowTemperatureRadiantElectric.empty?
|
308
|
+
equip = equip.to_ZoneHVACLowTemperatureRadiantElectric.get
|
309
|
+
sched = equip.heatingSetpointTemperatureSchedule
|
310
|
+
end
|
311
|
+
|
312
|
+
unless equip.to_ZoneHVACLowTempRadiantConstFlow.empty?
|
313
|
+
equip = equip.to_ZoneHVACLowTempRadiantConstFlow.get
|
314
|
+
coil = equip.heatingCoil
|
315
|
+
|
316
|
+
unless coil.to_CoilHeatingLowTempRadiantConstFlow.empty?
|
317
|
+
coil = coil.to_CoilHeatingLowTempRadiantConstFlow.get
|
318
|
+
|
319
|
+
unless coil.heatingHighControlTemperatureSchedule.empty?
|
320
|
+
sched = c.heatingHighControlTemperatureSchedule.get
|
321
|
+
end
|
322
|
+
end
|
323
|
+
end
|
324
|
+
|
325
|
+
unless equip.to_ZoneHVACLowTempRadiantVarFlow.empty?
|
326
|
+
equip = equip.to_ZoneHVACLowTempRadiantVarFlow.get
|
327
|
+
coil = equip.heatingCoil
|
328
|
+
|
329
|
+
unless coil.to_CoilHeatingLowTempRadiantVarFlow.empty?
|
330
|
+
coil = coil.to_CoilHeatingLowTempRadiantVarFlow.get
|
331
|
+
|
332
|
+
unless coil.heatingControlTemperatureSchedule.empty?
|
333
|
+
sched = coil.heatingControlTemperatureSchedule.get
|
334
|
+
end
|
335
|
+
end
|
336
|
+
end
|
337
|
+
|
338
|
+
next unless sched
|
339
|
+
|
340
|
+
unless sched.to_ScheduleRuleset.empty?
|
341
|
+
sched = sched.to_ScheduleRuleset.get
|
342
|
+
max = scheduleRulesetMinMax(sched)[:max]
|
343
|
+
|
344
|
+
if max
|
345
|
+
res[:spt] = max unless res[:spt]
|
346
|
+
res[:spt] = max if res[:spt] < max
|
347
|
+
end
|
348
|
+
end
|
349
|
+
|
350
|
+
unless sched.to_ScheduleConstant.empty?
|
351
|
+
sched = sched.to_ScheduleConstant.get
|
352
|
+
max = scheduleConstantMinMax(sched)[:max]
|
353
|
+
|
354
|
+
if max
|
355
|
+
res[:spt] = max unless res[:spt]
|
356
|
+
res[:spt] = max if res[:spt] < max
|
357
|
+
end
|
358
|
+
end
|
359
|
+
|
360
|
+
unless sched.to_ScheduleCompact.empty?
|
361
|
+
sched = sched.to_ScheduleCompact.get
|
362
|
+
max = scheduleCompactMinMax(sched)[:max]
|
363
|
+
|
364
|
+
if max
|
365
|
+
res[:spt] = max unless res[:spt]
|
366
|
+
res[:spt] = max if res[:spt] < max
|
367
|
+
end
|
368
|
+
end
|
369
|
+
end
|
370
|
+
|
371
|
+
return res if zone.thermostat.empty?
|
372
|
+
tstat = zone.thermostat.get
|
373
|
+
res[:spt] = nil
|
374
|
+
|
375
|
+
unless tstat.to_ThermostatSetpointDualSetpoint.empty? &&
|
376
|
+
tstat.to_ZoneControlThermostatStagedDualSetpoint.empty?
|
377
|
+
|
378
|
+
unless tstat.to_ThermostatSetpointDualSetpoint.empty?
|
379
|
+
tstat = tstat.to_ThermostatSetpointDualSetpoint.get
|
380
|
+
else
|
381
|
+
tstat = tstat.to_ZoneControlThermostatStagedDualSetpoint.get
|
382
|
+
end
|
383
|
+
|
384
|
+
unless tstat.heatingSetpointTemperatureSchedule.empty?
|
385
|
+
res[:dual] = true
|
386
|
+
sched = tstat.heatingSetpointTemperatureSchedule.get
|
387
|
+
|
388
|
+
unless sched.to_ScheduleRuleset.empty?
|
389
|
+
sched = sched.to_ScheduleRuleset.get
|
390
|
+
max = scheduleRulesetMinMax(sched)[:max]
|
391
|
+
|
392
|
+
if max
|
393
|
+
res[:spt] = max unless res[:spt]
|
394
|
+
res[:spt] = max if res[:spt] < max
|
395
|
+
end
|
396
|
+
|
397
|
+
dd = sched.winterDesignDaySchedule
|
398
|
+
|
399
|
+
unless dd.values.empty?
|
400
|
+
res[:spt] = dd.values.max unless res[:spt]
|
401
|
+
res[:spt] = dd.values.max if res[:spt] < dd.values.max
|
402
|
+
end
|
403
|
+
end
|
404
|
+
|
405
|
+
unless sched.to_ScheduleConstant.empty?
|
406
|
+
sched = sched.to_ScheduleConstant.get
|
407
|
+
max = scheduleConstantMinMax(sched)[:max]
|
408
|
+
|
409
|
+
if max
|
410
|
+
res[:spt] = max unless res[:spt]
|
411
|
+
res[:spt] = max if res[:spt] < max
|
412
|
+
end
|
413
|
+
end
|
414
|
+
|
415
|
+
unless sched.to_ScheduleCompact.empty?
|
416
|
+
sched = sched.to_ScheduleCompact.get
|
417
|
+
max = scheduleCompactMinMax(sched)[:max]
|
418
|
+
|
419
|
+
if max
|
420
|
+
res[:spt] = max unless res[:spt]
|
421
|
+
res[:spt] = max if res[:spt] < max
|
422
|
+
end
|
423
|
+
end
|
424
|
+
|
425
|
+
unless sched.to_ScheduleYear.empty?
|
426
|
+
sched = sched.to_ScheduleYear.get
|
427
|
+
|
428
|
+
sched.getScheduleWeeks.each do |week|
|
429
|
+
next if week.winterDesignDaySchedule.empty?
|
430
|
+
dd = week.winterDesignDaySchedule.get
|
431
|
+
next unless dd.values.empty?
|
432
|
+
|
433
|
+
res[:spt] = dd.values.max unless res[:spt]
|
434
|
+
res[:spt] = dd.values.max if res[:spt] < dd.values.max
|
435
|
+
end
|
436
|
+
end
|
437
|
+
end
|
438
|
+
end
|
439
|
+
|
440
|
+
res
|
441
|
+
end
|
442
|
+
|
443
|
+
##
|
444
|
+
# Validate if model has zones with valid heating temperature setpoints.
|
445
|
+
#
|
446
|
+
# @param model [OpenStudio::Model::Model] a model
|
447
|
+
#
|
448
|
+
# @return [Bool] true if valid heating temperature setpoints
|
449
|
+
# @return [Bool] false if invalid input
|
450
|
+
def heatingTemperatureSetpoints?(model = nil)
|
451
|
+
mth = "OSut::#{__callee__}"
|
452
|
+
cl = OpenStudio::Model::Model
|
453
|
+
|
454
|
+
return mismatch("model", model, cl, mth, DBG, false) unless model.is_a?(cl)
|
455
|
+
|
456
|
+
model.getThermalZones.each do |zone|
|
457
|
+
return true if maxHeatScheduledSetpoint(zone)[:spt]
|
458
|
+
end
|
459
|
+
|
460
|
+
false
|
461
|
+
end
|
462
|
+
|
463
|
+
##
|
464
|
+
# Return min zone cooling temperature schedule setpoint [°C] and whether
|
465
|
+
# zone has active dual setpoint thermostat.
|
466
|
+
#
|
467
|
+
# @param zone [OpenStudio::Model::ThermalZone] a thermal zone
|
468
|
+
#
|
469
|
+
# @return [Hash] spt: (Float), dual: (Bool)
|
470
|
+
# @return [Hash] spt: nil, dual: false (if invalid input)
|
471
|
+
def minCoolScheduledSetpoint(zone = nil)
|
472
|
+
# Largely inspired from Parker & Marrec's "thermal_zone_cooled?" procedure.
|
473
|
+
#
|
474
|
+
# github.com/NREL/openstudio-standards/blob/
|
475
|
+
# 99cf713750661fe7d2082739f251269c2dfd9140/lib/openstudio-standards/
|
476
|
+
# standards/Standards.ThermalZone.rb#L1058
|
477
|
+
mth = "OSut::#{__callee__}"
|
478
|
+
cl = OpenStudio::Model::ThermalZone
|
479
|
+
res = { spt: nil, dual: false }
|
480
|
+
|
481
|
+
return invalid("zone", mth, 1, DBG, res) unless zone.respond_to?(NS)
|
482
|
+
id = zone.nameString
|
483
|
+
return mismatch(id, zone, cl, mth, DBG, res) unless zone.is_a?(cl)
|
484
|
+
|
485
|
+
# Zone radiant cooling? Get schedule from radiant system.
|
486
|
+
zone.equipment.each do |equip|
|
487
|
+
sched = nil
|
488
|
+
|
489
|
+
unless equip.to_ZoneHVACLowTempRadiantConstFlow.empty?
|
490
|
+
equip = equip.to_ZoneHVACLowTempRadiantConstFlow.get
|
491
|
+
coil = equip.coolingCoil
|
492
|
+
|
493
|
+
unless coil.to_CoilCoolingLowTempRadiantConstFlow.empty?
|
494
|
+
coil = coil.to_CoilCoolingLowTempRadiantConstFlow.get
|
495
|
+
|
496
|
+
unless coil.coolingLowControlTemperatureSchedule.empty?
|
497
|
+
sched = coil.coolingLowControlTemperatureSchedule.get
|
498
|
+
end
|
499
|
+
end
|
500
|
+
end
|
501
|
+
|
502
|
+
unless equip.to_ZoneHVACLowTempRadiantVarFlow.empty?
|
503
|
+
equip = equip.to_ZoneHVACLowTempRadiantVarFlow.get
|
504
|
+
coil = equip.coolingCoil
|
505
|
+
|
506
|
+
unless coil.to_CoilCoolingLowTempRadiantVarFlow.empty?
|
507
|
+
coil = coil.to_CoilCoolingLowTempRadiantVarFlow.get
|
508
|
+
|
509
|
+
unless coil.coolingControlTemperatureSchedule.empty?
|
510
|
+
sched = coil.coolingControlTemperatureSchedule.get
|
511
|
+
end
|
512
|
+
end
|
513
|
+
end
|
514
|
+
|
515
|
+
next unless sched
|
516
|
+
|
517
|
+
unless sched.to_ScheduleRuleset.empty?
|
518
|
+
sched = sched.to_ScheduleRuleset.get
|
519
|
+
min = scheduleRulesetMinMax(sched)[:min]
|
520
|
+
|
521
|
+
if min
|
522
|
+
res[:spt] = min unless res[:spt]
|
523
|
+
res[:spt] = min if res[:spt] > min
|
524
|
+
end
|
525
|
+
end
|
526
|
+
|
527
|
+
unless sched.to_ScheduleConstant.empty?
|
528
|
+
sched = sched.to_ScheduleConstant.get
|
529
|
+
min = scheduleConstantMinMax(sched)[:min]
|
530
|
+
|
531
|
+
if min
|
532
|
+
res[:spt] = min unless res[:spt]
|
533
|
+
res[:spt] = min if res[:spt] > min
|
534
|
+
end
|
535
|
+
end
|
536
|
+
|
537
|
+
unless sched.to_ScheduleCompact.empty?
|
538
|
+
sched = sched.to_ScheduleCompact.get
|
539
|
+
min = scheduleCompactMinMax(sched)[:min]
|
540
|
+
|
541
|
+
if min
|
542
|
+
res[:spt] = min unless res[:spt]
|
543
|
+
res[:spt] = min if res[:spt] > min
|
544
|
+
end
|
545
|
+
end
|
546
|
+
end
|
547
|
+
|
548
|
+
return res if zone.thermostat.empty?
|
549
|
+
tstat = zone.thermostat.get
|
550
|
+
res[:spt] = nil
|
551
|
+
|
552
|
+
unless tstat.to_ThermostatSetpointDualSetpoint.empty? &&
|
553
|
+
tstat.to_ZoneControlThermostatStagedDualSetpoint.empty?
|
554
|
+
|
555
|
+
unless tstat.to_ThermostatSetpointDualSetpoint.empty?
|
556
|
+
tstat = tstat.to_ThermostatSetpointDualSetpoint.get
|
557
|
+
else
|
558
|
+
tstat = tstat.to_ZoneControlThermostatStagedDualSetpoint.get
|
559
|
+
end
|
560
|
+
|
561
|
+
unless tstat.coolingSetpointTemperatureSchedule.empty?
|
562
|
+
res[:dual] = true
|
563
|
+
sched = tstat.coolingSetpointTemperatureSchedule.get
|
564
|
+
|
565
|
+
unless sched.to_ScheduleRuleset.empty?
|
566
|
+
sched = sched.to_ScheduleRuleset.get
|
567
|
+
min = scheduleRulesetMinMax(sched)[:min]
|
568
|
+
|
569
|
+
if min
|
570
|
+
res[:spt] = min unless res[:spt]
|
571
|
+
res[:spt] = min if res[:spt] > min
|
572
|
+
end
|
573
|
+
|
574
|
+
dd = sched.summerDesignDaySchedule
|
575
|
+
|
576
|
+
unless dd.values.empty?
|
577
|
+
res[:spt] = dd.values.min unless res[:spt]
|
578
|
+
res[:spt] = dd.values.min if res[:spt] > dd.values.min
|
579
|
+
end
|
580
|
+
end
|
581
|
+
|
582
|
+
unless sched.to_ScheduleConstant.empty?
|
583
|
+
sched = sched.to_ScheduleConstant.get
|
584
|
+
min = scheduleConstantMinMax(sched)[:min]
|
585
|
+
|
586
|
+
if min
|
587
|
+
res[:spt] = min unless res[:spt]
|
588
|
+
res[:spt] = min if res[:spt] > min
|
589
|
+
end
|
590
|
+
end
|
591
|
+
|
592
|
+
unless sched.to_ScheduleCompact.empty?
|
593
|
+
sched = sched.to_ScheduleCompact.get
|
594
|
+
min = scheduleCompactMinMax(sched)[:min]
|
595
|
+
|
596
|
+
if min
|
597
|
+
res[:spt] = min unless res[:spt]
|
598
|
+
res[:spt] = min if res[:spt] > min
|
599
|
+
end
|
600
|
+
end
|
601
|
+
|
602
|
+
unless sched.to_ScheduleYear.empty?
|
603
|
+
sched = sched.to_ScheduleYear.get
|
604
|
+
|
605
|
+
sched.getScheduleWeeks.each do |week|
|
606
|
+
next if week.summerDesignDaySchedule.empty?
|
607
|
+
dd = week.summerDesignDaySchedule.get
|
608
|
+
next unless dd.values.empty?
|
609
|
+
|
610
|
+
res[:spt] = dd.values.min unless res[:spt]
|
611
|
+
res[:spt] = dd.values.min if res[:spt] > dd.values.min
|
612
|
+
end
|
613
|
+
end
|
614
|
+
end
|
615
|
+
end
|
616
|
+
|
617
|
+
res
|
618
|
+
end
|
619
|
+
|
620
|
+
##
|
621
|
+
# Validate if model has zones with valid cooling temperature setpoints.
|
622
|
+
#
|
623
|
+
# @param model [OpenStudio::Model::Model] a model
|
624
|
+
#
|
625
|
+
# @return [Bool] true if valid cooling temperature setpoints
|
626
|
+
# @return [Bool] false if invalid input
|
627
|
+
def coolingTemperatureSetpoints?(model = nil)
|
628
|
+
mth = "OSut::#{__callee__}"
|
629
|
+
cl = OpenStudio::Model::Model
|
630
|
+
|
631
|
+
return mismatch("model", model, cl, mth, DBG, false) unless model.is_a?(cl)
|
632
|
+
|
633
|
+
model.getThermalZones.each do |zone|
|
634
|
+
return true if minCoolScheduledSetpoint(zone)[:spt]
|
635
|
+
end
|
636
|
+
|
637
|
+
false
|
638
|
+
end
|
639
|
+
|
640
|
+
##
|
641
|
+
# Validate if model has zones with HVAC air loops.
|
642
|
+
#
|
643
|
+
# @param model [OpenStudio::Model::Model] a model
|
644
|
+
#
|
645
|
+
# @return [Bool] true if model has one or more HVAC air loops
|
646
|
+
# @return [Bool] false if invalid input
|
647
|
+
def airLoopsHVAC?(model = nil)
|
648
|
+
mth = "OSut::#{__callee__}"
|
649
|
+
cl = OpenStudio::Model::Model
|
650
|
+
|
651
|
+
return mismatch("model", model, cl, mth, DBG, false) unless model.is_a?(cl)
|
652
|
+
|
653
|
+
model.getThermalZones.each do |zone|
|
654
|
+
next if zone.canBePlenum
|
655
|
+
return true unless zone.airLoopHVACs.empty?
|
656
|
+
return true if zone.isPlenum
|
657
|
+
end
|
658
|
+
|
659
|
+
false
|
660
|
+
end
|
661
|
+
|
662
|
+
##
|
663
|
+
# Validate whether space should be processed as a plenum.
|
664
|
+
#
|
665
|
+
# @param space [OpenStudio::Model::Space] a space
|
666
|
+
# @param loops [Bool] true if model has airLoopHVAC object(s)
|
667
|
+
# @param setpoints [Bool] true if model has valid temperature setpoints
|
668
|
+
#
|
669
|
+
# @return [Bool] true if should be tagged as plenum
|
670
|
+
# @return [Bool] false if invalid input
|
671
|
+
def plenum?(space = nil, loops = nil, setpoints = nil)
|
672
|
+
# Largely inspired from NREL's "space_plenum?" procedure:
|
673
|
+
#
|
674
|
+
# github.com/NREL/openstudio-standards/blob/
|
675
|
+
# 58964222d25783e9da4ae292e375fb0d5c902aa5/lib/openstudio-standards/
|
676
|
+
# standards/Standards.Space.rb#L1384
|
677
|
+
#
|
678
|
+
# A space may be tagged as a plenum if:
|
679
|
+
#
|
680
|
+
# CASE A: its zone's "isPlenum" == true (SDK method) for a fully-developed
|
681
|
+
# OpenStudio model (complete with HVAC air loops); OR
|
682
|
+
#
|
683
|
+
# CASE B: (IN ABSENCE OF HVAC AIRLOOPS) if it's excluded from a building's
|
684
|
+
# total floor area yet linked to a zone holding an 'inactive'
|
685
|
+
# thermostat, i.e. can't extract valid setpoints; OR
|
686
|
+
#
|
687
|
+
# CASE C: (IN ABSENCE OF HVAC AIRLOOPS & VALID SETPOINTS) it has "plenum"
|
688
|
+
# (case insensitive) as a spacetype (or as a spacetype's
|
689
|
+
# 'standards spacetype').
|
690
|
+
mth = "OSut::#{__callee__}"
|
691
|
+
cl = OpenStudio::Model::Space
|
692
|
+
|
693
|
+
return invalid("space", mth, 1, DBG, false) unless space.respond_to?(NS)
|
694
|
+
id = space.nameString
|
695
|
+
return mismatch(id, space, cl, mth, DBG, false) unless space.is_a?(cl)
|
696
|
+
valid = loops == true || loops == false
|
697
|
+
return invalid("loops", mth, 2, DBG, false) unless valid
|
698
|
+
valid = setpoints == true || setpoints == false
|
699
|
+
return invalid("setpoints", mth, 3, DBG, false) unless valid
|
700
|
+
|
701
|
+
unless space.thermalZone.empty?
|
702
|
+
zone = space.thermalZone.get
|
703
|
+
return zone.isPlenum if loops # A
|
704
|
+
|
705
|
+
if setpoints
|
706
|
+
heat = maxHeatScheduledSetpoint(zone)
|
707
|
+
cool = minCoolScheduledSetpoint(zone)
|
708
|
+
return false if heat[:spt] || cool[:spt] # directly conditioned
|
709
|
+
return heat[:dual] || cool[:dual] unless space.partofTotalFloorArea # B
|
710
|
+
return false
|
711
|
+
end
|
712
|
+
end
|
713
|
+
|
714
|
+
unless space.spaceType.empty?
|
715
|
+
type = space.spaceType.get
|
716
|
+
return type.nameString.downcase == "plenum" # C
|
717
|
+
|
718
|
+
unless type.standardsSpaceType.empty?
|
719
|
+
type = type.standardsSpaceType.get
|
720
|
+
return type.downcase == "plenum" # C
|
721
|
+
end
|
722
|
+
end
|
723
|
+
|
724
|
+
false
|
725
|
+
end
|
726
|
+
|
727
|
+
##
|
728
|
+
# Generate an HVAC availability schedule.
|
729
|
+
#
|
730
|
+
# @param model [OpenStudio::Model::Model] a model
|
731
|
+
# @param avl [String] seasonal availability choice (optional, default "ON")
|
732
|
+
#
|
733
|
+
# @return [OpenStudio::Model::Schedule] HVAC availability sched
|
734
|
+
# @return [NilClass] if invalid input
|
735
|
+
def availabilitySchedule(model = nil, avl = "")
|
736
|
+
mth = "OSut::#{__callee__}"
|
737
|
+
cl = OpenStudio::Model::Model
|
738
|
+
limits = nil
|
739
|
+
|
740
|
+
return mismatch("model", model, cl, mth) unless model.is_a?(cl)
|
741
|
+
return invalid("availability", avl, 2, mth) unless avl.respond_to?(:to_s)
|
742
|
+
|
743
|
+
# Either fetch availability ScheduleTypeLimits object, or create one.
|
744
|
+
model.getScheduleTypeLimitss.each do |l|
|
745
|
+
break if limits
|
746
|
+
next if l.lowerLimitValue.empty?
|
747
|
+
next if l.upperLimitValue.empty?
|
748
|
+
next if l.numericType.empty?
|
749
|
+
next unless l.lowerLimitValue.get.to_i == 0
|
750
|
+
next unless l.upperLimitValue.get.to_i == 1
|
751
|
+
next unless l.numericType.get.downcase == "discrete"
|
752
|
+
next unless l.unitType.downcase == "availability"
|
753
|
+
next unless l.nameString.downcase == "hvac operation scheduletypelimits"
|
754
|
+
limits = l
|
755
|
+
end
|
756
|
+
|
757
|
+
unless limits
|
758
|
+
limits = OpenStudio::Model::ScheduleTypeLimits.new(model)
|
759
|
+
limits.setName("HVAC Operation ScheduleTypeLimits")
|
760
|
+
limits.setLowerLimitValue(0)
|
761
|
+
limits.setUpperLimitValue(1)
|
762
|
+
limits.setNumericType("Discrete")
|
763
|
+
limits.setUnitType("Availability")
|
764
|
+
end
|
765
|
+
|
766
|
+
time = OpenStudio::Time.new(0,24)
|
767
|
+
secs = time.totalSeconds
|
768
|
+
on = OpenStudio::Model::ScheduleDay.new(model, 1)
|
769
|
+
off = OpenStudio::Model::ScheduleDay.new(model, 0)
|
770
|
+
|
771
|
+
# Seasonal availability start/end dates.
|
772
|
+
year = model.yearDescription
|
773
|
+
return empty("yearDescription", mth, ERR) if year.empty?
|
774
|
+
year = year.get
|
775
|
+
may01 = year.makeDate(OpenStudio::MonthOfYear.new("May"), 1)
|
776
|
+
oct31 = year.makeDate(OpenStudio::MonthOfYear.new("Oct"), 31)
|
777
|
+
|
778
|
+
case avl.to_s.downcase
|
779
|
+
when "winter" # available from November 1 to April 30 (6 months)
|
780
|
+
val = 1
|
781
|
+
sch = off
|
782
|
+
nom = "WINTER Availability SchedRuleset"
|
783
|
+
dft = "WINTER Availability dftDaySched"
|
784
|
+
tag = "May-Oct WINTER Availability SchedRule"
|
785
|
+
day = "May-Oct WINTER SchedRule Day"
|
786
|
+
when "summer" # available from May 1 to October 31 (6 months)
|
787
|
+
val = 0
|
788
|
+
sch = on
|
789
|
+
nom = "SUMMER Availability SchedRuleset"
|
790
|
+
dft = "SUMMER Availability dftDaySched"
|
791
|
+
tag = "May-Oct SUMMER Availability SchedRule"
|
792
|
+
day = "May-Oct SUMMER SchedRule Day"
|
793
|
+
when "off" # never available
|
794
|
+
val = 0
|
795
|
+
sch = on
|
796
|
+
nom = "OFF Availability SchedRuleset"
|
797
|
+
dft = "OFF Availability dftDaySched"
|
798
|
+
tag = ""
|
799
|
+
day = ""
|
800
|
+
else # always available
|
801
|
+
val = 1
|
802
|
+
sch = on
|
803
|
+
nom = "ON Availability SchedRuleset"
|
804
|
+
dft = "ON Availability dftDaySched"
|
805
|
+
tag = ""
|
806
|
+
day = ""
|
807
|
+
end
|
808
|
+
|
809
|
+
# Fetch existing schedule.
|
810
|
+
ok = true
|
811
|
+
schedule = model.getScheduleByName(nom)
|
812
|
+
|
813
|
+
unless schedule.empty?
|
814
|
+
schedule = schedule.get.to_ScheduleRuleset
|
815
|
+
|
816
|
+
unless schedule.empty?
|
817
|
+
schedule = schedule.get
|
818
|
+
default = schedule.defaultDaySchedule
|
819
|
+
ok = ok && default.nameString == dft
|
820
|
+
ok = ok && default.times.size == 1
|
821
|
+
ok = ok && default.values.size == 1
|
822
|
+
ok = ok && default.times.first == time
|
823
|
+
ok = ok && default.values.first == val
|
824
|
+
rules = schedule.scheduleRules
|
825
|
+
ok = ok && (rules.size == 0 || rules.size == 1)
|
826
|
+
|
827
|
+
if rules.size == 1
|
828
|
+
rule = rules.first
|
829
|
+
ok = ok && rule.nameString == tag
|
830
|
+
ok = ok && !rule.startDate.empty?
|
831
|
+
ok = ok && !rule.endDate.empty?
|
832
|
+
ok = ok && rule.startDate.get == may01
|
833
|
+
ok = ok && rule.endDate.get == oct31
|
834
|
+
ok = ok && rule.applyAllDays
|
835
|
+
|
836
|
+
d = rule.daySchedule
|
837
|
+
ok = ok && d.nameString == day
|
838
|
+
ok = ok && d.times.size == 1
|
839
|
+
ok = ok && d.values.size == 1
|
840
|
+
ok = ok && d.times.first.totalSeconds == secs
|
841
|
+
ok = ok && d.values.first.to_i != val
|
842
|
+
end
|
843
|
+
|
844
|
+
return schedule if ok
|
845
|
+
end
|
846
|
+
end
|
847
|
+
|
848
|
+
schedule = OpenStudio::Model::ScheduleRuleset.new(model)
|
849
|
+
schedule.setName(nom)
|
850
|
+
ok = schedule.setScheduleTypeLimits(limits)
|
851
|
+
log(ERR, "'#{nom}': Can't set schedule type limits (#{mth})") unless ok
|
852
|
+
return nil unless ok
|
853
|
+
ok = schedule.defaultDaySchedule.addValue(time, val)
|
854
|
+
log(ERR, "'#{nom}': Can't set default day schedule (#{mth})") unless ok
|
855
|
+
return nil unless ok
|
856
|
+
schedule.defaultDaySchedule.setName(dft)
|
857
|
+
|
858
|
+
unless tag.empty?
|
859
|
+
rule = OpenStudio::Model::ScheduleRule.new(schedule, sch)
|
860
|
+
rule.setName(tag)
|
861
|
+
ok = rule.setStartDate(may01)
|
862
|
+
log(ERR, "'#{tag}': Can't set start date (#{mth})") unless ok
|
863
|
+
return nil unless ok
|
864
|
+
ok = rule.setEndDate(oct31)
|
865
|
+
log(ERR, "'#{tag}': Can't set end date (#{mth})") unless ok
|
866
|
+
return nil unless ok
|
867
|
+
ok = rule.setApplyAllDays(true)
|
868
|
+
log(ERR, "'#{tag}': Can't apply to all days (#{mth})") unless ok
|
869
|
+
return nil unless ok
|
870
|
+
rule.daySchedule.setName(day)
|
871
|
+
end
|
872
|
+
|
873
|
+
schedule
|
874
|
+
end
|
875
|
+
|
876
|
+
##
|
877
|
+
# Validate if default construction set holds a base ground construction.
|
878
|
+
#
|
879
|
+
# @param set [OpenStudio::Model::DefaultConstructionSet] a default set
|
880
|
+
# @param bse [OpensStudio::Model::ConstructionBase] a construction base
|
881
|
+
# @param gr [Bool] true if ground-facing surface
|
882
|
+
# @param ex [Bool] true if exterior-facing surface
|
883
|
+
# @param typ [String] a surface type
|
884
|
+
#
|
885
|
+
# @return [Bool] true if default construction set holds construction
|
886
|
+
# @return [Bool] false if invalid input
|
887
|
+
def holdsConstruction?(set = nil, bse = nil, gr = false, ex = false, typ = "")
|
888
|
+
mth = "OSut::#{__callee__}"
|
889
|
+
cl1 = OpenStudio::Model::DefaultConstructionSet
|
890
|
+
cl2 = OpenStudio::Model::ConstructionBase
|
891
|
+
|
892
|
+
return invalid("set", mth, 1, DBG, false) unless set.respond_to?(NS)
|
893
|
+
id = set.nameString
|
894
|
+
return mismatch(id, set, cl1, mth, DBG, false) unless set.is_a?(cl1)
|
895
|
+
return invalid("base", mth, 2, DBG, false) unless bse.respond_to?(NS)
|
896
|
+
id = bse.nameString
|
897
|
+
return mismatch(id, bse, cl2, mth, DBG, false) unless bse.is_a?(cl2)
|
898
|
+
valid = gr == true || gr == false
|
899
|
+
return invalid("ground", mth, 3, DBG, false) unless valid
|
900
|
+
valid = ex == true || ex == false
|
901
|
+
return invalid("exterior", mth, 4, DBG, false) unless valid
|
902
|
+
valid = typ.respond_to?(:to_s)
|
903
|
+
return invalid("surface typ", mth, 4, DBG, false) unless valid
|
904
|
+
type = typ.to_s.downcase
|
905
|
+
valid = type == "floor" || type == "wall" || type == "roofceiling"
|
906
|
+
return invalid("surface type", mth, 5, DBG, false) unless valid
|
907
|
+
|
908
|
+
constructions = nil
|
909
|
+
|
910
|
+
if gr
|
911
|
+
unless set.defaultGroundContactSurfaceConstructions.empty?
|
912
|
+
constructions = set.defaultGroundContactSurfaceConstructions.get
|
913
|
+
end
|
914
|
+
elsif ex
|
915
|
+
unless set.defaultExteriorSurfaceConstructions.empty?
|
916
|
+
constructions = set.defaultExteriorSurfaceConstructions.get
|
917
|
+
end
|
918
|
+
else
|
919
|
+
unless set.defaultInteriorSurfaceConstructions.empty?
|
920
|
+
constructions = set.defaultInteriorSurfaceConstructions.get
|
921
|
+
end
|
922
|
+
end
|
923
|
+
|
924
|
+
return false unless constructions
|
925
|
+
|
926
|
+
case type
|
927
|
+
when "roofceiling"
|
928
|
+
unless constructions.roofCeilingConstruction.empty?
|
929
|
+
construction = constructions.roofCeilingConstruction.get
|
930
|
+
return true if construction == bse
|
931
|
+
end
|
932
|
+
when "floor"
|
933
|
+
unless constructions.floorConstruction.empty?
|
934
|
+
construction = constructions.floorConstruction.get
|
935
|
+
return true if construction == bse
|
936
|
+
end
|
937
|
+
else
|
938
|
+
unless constructions.wallConstruction.empty?
|
939
|
+
construction = constructions.wallConstruction.get
|
940
|
+
return true if construction == bse
|
941
|
+
end
|
942
|
+
end
|
943
|
+
|
944
|
+
false
|
945
|
+
end
|
946
|
+
|
947
|
+
##
|
948
|
+
# Return a surface's default construction set.
|
949
|
+
#
|
950
|
+
# @param model [OpenStudio::Model::Model] a model
|
951
|
+
# @param s [OpenStudio::Model::Surface] a surface
|
952
|
+
#
|
953
|
+
# @return [OpenStudio::Model::DefaultConstructionSet] default set
|
954
|
+
# @return [NilClass] if invalid input
|
955
|
+
def defaultConstructionSet(model = nil, s = nil)
|
956
|
+
mth = "OSut::#{__callee__}"
|
957
|
+
cl1 = OpenStudio::Model::Model
|
958
|
+
cl2 = OpenStudio::Model::Surface
|
959
|
+
|
960
|
+
return mismatch("model", model, cl1, mth) unless model.is_a?(cl1)
|
961
|
+
return invalid("s", mth, 2) unless s.respond_to?(NS)
|
962
|
+
id = s.nameString
|
963
|
+
return mismatch(id, s, cl2, mth) unless s.is_a?(cl2)
|
964
|
+
|
965
|
+
ok = s.isConstructionDefaulted
|
966
|
+
log(ERR, "'#{id}' construction not defaulted (#{mth})") unless ok
|
967
|
+
return nil unless ok
|
968
|
+
return empty("'#{id}' construction", mth, ERR) if s.construction.empty?
|
969
|
+
base = s.construction.get
|
970
|
+
return empty("'#{id}' space", mth, ERR) if s.space.empty?
|
971
|
+
space = s.space.get
|
972
|
+
type = s.surfaceType
|
973
|
+
ground = false
|
974
|
+
exterior = false
|
975
|
+
|
976
|
+
if s.isGroundSurface
|
977
|
+
ground = true
|
978
|
+
elsif s.outsideBoundaryCondition.downcase == "outdoors"
|
979
|
+
exterior = true
|
980
|
+
end
|
981
|
+
|
982
|
+
unless space.defaultConstructionSet.empty?
|
983
|
+
set = space.defaultConstructionSet.get
|
984
|
+
return set if holdsConstruction?(set, base, ground, exterior, type)
|
985
|
+
end
|
986
|
+
|
987
|
+
unless space.spaceType.empty?
|
988
|
+
spacetype = space.spaceType.get
|
989
|
+
|
990
|
+
unless spacetype.defaultConstructionSet.empty?
|
991
|
+
set = spacetype.defaultConstructionSet.get
|
992
|
+
return set if holdsConstruction?(set, base, ground, exterior, type)
|
993
|
+
end
|
994
|
+
end
|
995
|
+
|
996
|
+
unless space.buildingStory.empty?
|
997
|
+
story = space.buildingStory.get
|
998
|
+
|
999
|
+
unless story.defaultConstructionSet.empty?
|
1000
|
+
set = story.defaultConstructionSet.get
|
1001
|
+
return set if holdsConstruction?(set, base, ground, exterior, type)
|
1002
|
+
end
|
1003
|
+
end
|
1004
|
+
|
1005
|
+
building = model.getBuilding
|
1006
|
+
|
1007
|
+
unless building.defaultConstructionSet.empty?
|
1008
|
+
set = building.defaultConstructionSet.get
|
1009
|
+
return set if holdsConstruction?(set, base, ground, exterior, type)
|
1010
|
+
end
|
1011
|
+
|
1012
|
+
nil
|
1013
|
+
end
|
1014
|
+
|
1015
|
+
##
|
1016
|
+
# Validate if every material in a layered construction is standard & opaque.
|
1017
|
+
#
|
1018
|
+
# @param lc [OpenStudio::LayeredConstruction] a layered construction
|
1019
|
+
#
|
1020
|
+
# @return [Bool] true if all layers are valid
|
1021
|
+
# @return [Bool] false if invalid input
|
1022
|
+
def standardOpaqueLayers?(lc = nil)
|
1023
|
+
mth = "OSut::#{__callee__}"
|
1024
|
+
cl = OpenStudio::Model::LayeredConstruction
|
1025
|
+
|
1026
|
+
return invalid("lc", mth, 1, DBG, false) unless lc.respond_to?(NS)
|
1027
|
+
return mismatch(lc.nameString, lc, cl, mth, DBG, false) unless lc.is_a?(cl)
|
1028
|
+
|
1029
|
+
lc.layers.each { |m| return false if m.to_StandardOpaqueMaterial.empty? }
|
1030
|
+
|
1031
|
+
true
|
1032
|
+
end
|
1033
|
+
|
1034
|
+
##
|
1035
|
+
# Total (standard opaque) layered construction thickness (in m).
|
1036
|
+
#
|
1037
|
+
# @param lc [OpenStudio::LayeredConstruction] a layered construction
|
1038
|
+
#
|
1039
|
+
# @return [Float] total layered construction thickness
|
1040
|
+
# @return [Float] 0 if invalid input
|
1041
|
+
def thickness(lc = nil)
|
1042
|
+
mth = "OSut::#{__callee__}"
|
1043
|
+
cl = OpenStudio::Model::LayeredConstruction
|
1044
|
+
|
1045
|
+
return invalid("lc", mth, 1, DBG, 0.0) unless lc.respond_to?(NS)
|
1046
|
+
id = lc.nameString
|
1047
|
+
return mismatch(id, lc, cl, mth, DBG, 0.0) unless lc.is_a?(cl)
|
1048
|
+
|
1049
|
+
ok = standardOpaqueLayers?(lc)
|
1050
|
+
log(ERR, "'#{id}' holds non-StandardOpaqueMaterial(s) (#{mth})") unless ok
|
1051
|
+
return 0.0 unless ok
|
1052
|
+
thickness = 0.0
|
1053
|
+
lc.layers.each { |m| thickness += m.thickness }
|
1054
|
+
|
1055
|
+
thickness
|
1056
|
+
end
|
1057
|
+
|
1058
|
+
##
|
1059
|
+
# Return total air film resistance for fenestration.
|
1060
|
+
#
|
1061
|
+
# @param usi [Float] a fenestrated construction's U-factor (W/m2•K)
|
1062
|
+
#
|
1063
|
+
# @return [Float] total air film resistance in m2•K/W (0.1216 if errors)
|
1064
|
+
def glazingAirFilmRSi(usi = 5.85)
|
1065
|
+
# The sum of thermal resistances of calculated exterior and interior film
|
1066
|
+
# coefficients under standard winter conditions are taken from:
|
1067
|
+
#
|
1068
|
+
# https://bigladdersoftware.com/epx/docs/9-6/engineering-reference/
|
1069
|
+
# window-calculation-module.html#simple-window-model
|
1070
|
+
#
|
1071
|
+
# These remain acceptable approximations for flat windows, yet likely
|
1072
|
+
# unsuitable for subsurfaces with curved or projecting shapes like domed
|
1073
|
+
# skylights. The solution here is considered an adequate fix for reporting,
|
1074
|
+
# awaiting eventual OpenStudio (and EnergyPlus) upgrades to report NFRC 100
|
1075
|
+
# (or ISO) air film resistances under standard winter conditions.
|
1076
|
+
#
|
1077
|
+
# For U-factors above 8.0 W/m2•K (or invalid input), the function returns
|
1078
|
+
# 0.1216 m2•K/W, which corresponds to a construction with a single glass
|
1079
|
+
# layer thickness of 2mm & k = ~0.6 W/m.K.
|
1080
|
+
#
|
1081
|
+
# The EnergyPlus Engineering calculations were designed for vertical windows
|
1082
|
+
# - not horizontal, slanted or domed surfaces - use with caution.
|
1083
|
+
mth = "OSut::#{__callee__}"
|
1084
|
+
cl = Numeric
|
1085
|
+
|
1086
|
+
return mismatch("usi", usi, cl, mth, DBG, 0.1216) unless usi.is_a?(cl)
|
1087
|
+
return invalid("usi", mth, 1, WRN, 0.1216) if usi > 8.0
|
1088
|
+
return negative("usi", mth, WRN, 0.1216) if usi < 0
|
1089
|
+
return zero("usi", mth, WRN, 0.1216) if usi.abs < TOL
|
1090
|
+
|
1091
|
+
rsi = 1 / (0.025342 * usi + 29.163853) # exterior film, next interior film
|
1092
|
+
|
1093
|
+
return rsi + 1 / (0.359073 * Math.log(usi) + 6.949915) if usi < 5.85
|
1094
|
+
return rsi + 1 / (1.788041 * usi - 2.886625)
|
1095
|
+
end
|
1096
|
+
|
1097
|
+
##
|
1098
|
+
# Return a construction's 'standard calc' thermal resistance (with air films).
|
1099
|
+
#
|
1100
|
+
# @param lc [OpenStudio::Model::LayeredConstruction] a layered construction
|
1101
|
+
# @param film [Float] thermal resistance of surface air films (m2•K/W)
|
1102
|
+
# @param t [Float] gas temperature (°C) (optional)
|
1103
|
+
#
|
1104
|
+
# @return [Float] calculated RSi at standard conditions (0 if error)
|
1105
|
+
def rsi(lc = nil, film = 0.0, t = 0.0)
|
1106
|
+
# This is adapted from BTAP's Material Module's "get_conductance" (P. Lopez)
|
1107
|
+
#
|
1108
|
+
# https://github.com/NREL/OpenStudio-Prototype-Buildings/blob/
|
1109
|
+
# c3d5021d8b7aef43e560544699fb5c559e6b721d/lib/btap/measures/
|
1110
|
+
# btap_equest_converter/envelope.rb#L122
|
1111
|
+
mth = "OSut::#{__callee__}"
|
1112
|
+
cl1 = OpenStudio::Model::LayeredConstruction
|
1113
|
+
cl2 = Numeric
|
1114
|
+
|
1115
|
+
return invalid("lc", mth, 1, DBG, 0.0) unless lc.respond_to?(NS)
|
1116
|
+
id = lc.nameString
|
1117
|
+
return mismatch(id, lc, cl1, mth, DBG, 0.0) unless lc.is_a?(cl1)
|
1118
|
+
return mismatch("film", film, cl2, mth, DBG, 0.0) unless film.is_a?(cl2)
|
1119
|
+
return mismatch("temp K", t, cl2, mth, DBG, 0.0) unless t.is_a?(cl2)
|
1120
|
+
t += 273.0 # °C to K
|
1121
|
+
return negative("temp K", mth, DBG, 0.0) if t < 0
|
1122
|
+
return negative("film", mth, DBG, 0.0) if film < 0
|
1123
|
+
|
1124
|
+
rsi = film
|
1125
|
+
|
1126
|
+
lc.layers.each do |m|
|
1127
|
+
# Fenestration materials first (ignoring shades, screens, etc.)
|
1128
|
+
empty = m.to_SimpleGlazing.empty?
|
1129
|
+
return 1 / m.to_SimpleGlazing.get.uFactor unless empty
|
1130
|
+
empty = m.to_StandardGlazing.empty?
|
1131
|
+
rsi += m.to_StandardGlazing.get.thermalResistance unless empty
|
1132
|
+
empty = m.to_RefractionExtinctionGlazing.empty?
|
1133
|
+
rsi += m.to_RefractionExtinctionGlazing.get.thermalResistance unless empty
|
1134
|
+
empty = m.to_Gas.empty?
|
1135
|
+
rsi += m.to_Gas.get.getThermalResistance(t) unless empty
|
1136
|
+
empty = m.to_GasMixture.empty?
|
1137
|
+
rsi += m.to_GasMixture.get.getThermalResistance(t) unless empty
|
1138
|
+
|
1139
|
+
# Opaque materials next.
|
1140
|
+
empty = m.to_StandardOpaqueMaterial.empty?
|
1141
|
+
rsi += m.to_StandardOpaqueMaterial.get.thermalResistance unless empty
|
1142
|
+
empty = m.to_MasslessOpaqueMaterial.empty?
|
1143
|
+
rsi += m.to_MasslessOpaqueMaterial.get.thermalResistance unless empty
|
1144
|
+
empty = m.to_RoofVegetation.empty?
|
1145
|
+
rsi += m.to_RoofVegetation.get.thermalResistance unless empty
|
1146
|
+
empty = m.to_AirGap.empty?
|
1147
|
+
rsi += m.to_AirGap.get.thermalResistance unless empty
|
1148
|
+
end
|
1149
|
+
|
1150
|
+
rsi
|
1151
|
+
end
|
1152
|
+
|
1153
|
+
##
|
1154
|
+
# Identify a layered construction's (opaque) insulating layer. The method
|
1155
|
+
# returns a 3-keyed hash ... :index (insulating layer index within layered
|
1156
|
+
# construction), :type (standard: or massless: material type), and
|
1157
|
+
# :r (material thermal resistance in m2•K/W).
|
1158
|
+
#
|
1159
|
+
# @param lc [OpenStudio::Model::LayeredConstruction] a layered construction
|
1160
|
+
#
|
1161
|
+
# @return [Hash] index: (Integer), type: (:standard or :massless), r: (Float)
|
1162
|
+
# @return [Hash] index: nil, type: nil, r: 0 (if invalid input)
|
1163
|
+
def insulatingLayer(lc = nil)
|
1164
|
+
mth = "OSut::#{__callee__}"
|
1165
|
+
cl = OpenStudio::Model::LayeredConstruction
|
1166
|
+
res = { index: nil, type: nil, r: 0.0 }
|
1167
|
+
i = 0 # iterator
|
1168
|
+
|
1169
|
+
return invalid("lc", mth, 1, DBG, res) unless lc.respond_to?(NS)
|
1170
|
+
id = lc.nameString
|
1171
|
+
return mismatch(id, lc, cl1, mth, DBG, res) unless lc.is_a?(cl)
|
1172
|
+
|
1173
|
+
lc.layers.each do |m|
|
1174
|
+
unless m.to_MasslessOpaqueMaterial.empty?
|
1175
|
+
m = m.to_MasslessOpaqueMaterial.get
|
1176
|
+
|
1177
|
+
if m.thermalResistance < 0.001 || m.thermalResistance < res[:r]
|
1178
|
+
i += 1
|
1179
|
+
next
|
1180
|
+
else
|
1181
|
+
res[:r ] = m.thermalResistance
|
1182
|
+
res[:index] = i
|
1183
|
+
res[:type ] = :massless
|
1184
|
+
end
|
1185
|
+
end
|
1186
|
+
|
1187
|
+
unless m.to_StandardOpaqueMaterial.empty?
|
1188
|
+
m = m.to_StandardOpaqueMaterial.get
|
1189
|
+
k = m.thermalConductivity
|
1190
|
+
d = m.thickness
|
1191
|
+
|
1192
|
+
if d < 0.003 || k > 3.0 || d / k < res[:r]
|
1193
|
+
i += 1
|
1194
|
+
next
|
1195
|
+
else
|
1196
|
+
res[:r ] = d / k
|
1197
|
+
res[:index] = i
|
1198
|
+
res[:type ] = :standard
|
1199
|
+
end
|
1200
|
+
end
|
1201
|
+
|
1202
|
+
i += 1
|
1203
|
+
end
|
1204
|
+
|
1205
|
+
res
|
1206
|
+
end
|
1207
|
+
|
1208
|
+
##
|
1209
|
+
# Return OpenStudio site/space transformation & rotation angle [0,2PI) rads.
|
1210
|
+
#
|
1211
|
+
# @param model [OpenStudio::Model::Model] a model
|
1212
|
+
# @param group [OpenStudio::Model::PlanarSurfaceGroup] a group
|
1213
|
+
#
|
1214
|
+
# @return [Hash] t: (OpenStudio::Transformation), r: Float
|
1215
|
+
# @return [Hash] t: nil, r: nil (if invalid input)
|
1216
|
+
def transforms(model = nil, group = nil)
|
1217
|
+
mth = "OSut::#{__callee__}"
|
1218
|
+
cl1 = OpenStudio::Model::Model
|
1219
|
+
cl2 = OpenStudio::Model::PlanarSurfaceGroup
|
1220
|
+
res = { t: nil, r: nil }
|
1221
|
+
|
1222
|
+
return mismatch("model", model, cl1, mth, DBG, res) unless model.is_a?(cl1)
|
1223
|
+
return invalid("group", mth, 2, DBG, res) unless group.respond_to?(NS)
|
1224
|
+
id = group.nameString
|
1225
|
+
return mismatch(id, group, cl2, mth, DBG, res) unless group.is_a?(cl2)
|
1226
|
+
|
1227
|
+
res[:t] = group.siteTransformation
|
1228
|
+
res[:r] = group.directionofRelativeNorth + model.getBuilding.northAxis
|
1229
|
+
|
1230
|
+
res
|
1231
|
+
end
|
1232
|
+
|
1233
|
+
##
|
1234
|
+
# Return a scalar product of an OpenStudio Vector3d.
|
1235
|
+
#
|
1236
|
+
# @param v [OpenStudio::Vector3d] a vector
|
1237
|
+
# @param m [Float] a scalar
|
1238
|
+
#
|
1239
|
+
# @return [OpenStudio::Vector3d] modified vector
|
1240
|
+
# @return [OpenStudio::Vector3d] provided (or empty) vector if invalid input
|
1241
|
+
def scalar(v = OpenStudio::Vector3d.new(0,0,0), m = 0)
|
1242
|
+
mth = "OSut::#{__callee__}"
|
1243
|
+
cl1 = OpenStudio::Vector3d
|
1244
|
+
cl2 = Numeric
|
1245
|
+
|
1246
|
+
return mismatch("vector", v, cl1, mth, DBG, v) unless v.is_a?(cl1)
|
1247
|
+
return mismatch("x", v.x, cl2, mth, DBG, v) unless v.x.respond_to?(:to_f)
|
1248
|
+
return mismatch("y", v.y, cl2, mth, DBG, v) unless v.y.respond_to?(:to_f)
|
1249
|
+
return mismatch("z", v.z, cl2, mth, DBG, v) unless v.z.respond_to?(:to_f)
|
1250
|
+
return mismatch("m", m, cl2, mth, DBG, v) unless m.respond_to?(:to_f)
|
1251
|
+
|
1252
|
+
OpenStudio::Vector3d.new(m * v.x, m * v.y, m * v.z)
|
1253
|
+
end
|
1254
|
+
|
1255
|
+
##
|
1256
|
+
# Flatten OpenStudio 3D points vs Z-axis (Z=0).
|
1257
|
+
#
|
1258
|
+
# @param pts [Array] an OpenStudio Point3D array/vector
|
1259
|
+
#
|
1260
|
+
# @return [Array] flattened OpenStudio 3D points
|
1261
|
+
def flatZ(pts = nil)
|
1262
|
+
mth = "OSut::#{__callee__}"
|
1263
|
+
cl1 = OpenStudio::Point3dVector
|
1264
|
+
cl2 = OpenStudio::Point3d
|
1265
|
+
v = OpenStudio::Point3dVector.new
|
1266
|
+
|
1267
|
+
valid = pts.is_a?(cl1) || pts.is_a?(Array)
|
1268
|
+
return mismatch("points", pts, cl1, mth, DBG, v) unless valid
|
1269
|
+
pts.each { |pt| mismatch("pt", pt, cl2, mth, ERR, v) unless pt.is_a?(cl2) }
|
1270
|
+
pts.each { |pt| v << OpenStudio::Point3d.new(pt.x, pt.y, 0) }
|
1271
|
+
|
1272
|
+
v
|
1273
|
+
end
|
1274
|
+
|
1275
|
+
##
|
1276
|
+
# Validate whether 1st OpenStudio convex polygon fits in 2nd convex polygon.
|
1277
|
+
#
|
1278
|
+
# @param p1 [OpenStudio::Point3dVector] or Point3D array of polygon #1
|
1279
|
+
# @param p2 [OpenStudio::Point3dVector] or Point3D array of polygon #2
|
1280
|
+
# @param id1 [String] polygon #1 identifier (optional)
|
1281
|
+
# @param id2 [String] polygon #2 identifier (optional)
|
1282
|
+
#
|
1283
|
+
# @return [Bool] true if 1st polygon fits entirely within the 2nd polygon
|
1284
|
+
# @return [Bool] false if invalid input
|
1285
|
+
def fits?(p1 = nil, p2 = nil, id1 = "", id2 = "")
|
1286
|
+
mth = "OSut::#{__callee__}"
|
1287
|
+
cl1 = OpenStudio::Point3dVector
|
1288
|
+
cl2 = OpenStudio::Point3d
|
1289
|
+
a = false
|
1290
|
+
|
1291
|
+
return invalid("id1", mth, 3, DBG, a) unless id1.respond_to?(:to_s)
|
1292
|
+
return invalid("id2", mth, 4, DBG, a) unless id2.respond_to?(:to_s)
|
1293
|
+
|
1294
|
+
i1 = id1.to_s
|
1295
|
+
i2 = id2.to_s
|
1296
|
+
i1 = "poly1" if i1.empty?
|
1297
|
+
i2 = "poly2" if i2.empty?
|
1298
|
+
|
1299
|
+
valid1 = p1.is_a?(cl1) || p1.is_a?(Array)
|
1300
|
+
valid2 = p2.is_a?(cl1) || p2.is_a?(Array)
|
1301
|
+
|
1302
|
+
return mismatch(i1, p1, cl1, mth, DBG, a) unless valid1
|
1303
|
+
return mismatch(i2, p2, cl1, mth, DBG, a) unless valid2
|
1304
|
+
return empty(i1, mth, ERR, a) if p1.empty?
|
1305
|
+
return empty(i2, mth, ERR, a) if p2.empty?
|
1306
|
+
|
1307
|
+
p1.each { |v| return mismatch(i1, v, cl2, mth, ERR, a) unless v.is_a?(cl2) }
|
1308
|
+
p2.each { |v| return mismatch(i2, v, cl2, mth, ERR, a) unless v.is_a?(cl2) }
|
1309
|
+
|
1310
|
+
# XY-plane transformation matrix ... needs to be clockwise for boost.
|
1311
|
+
ft = OpenStudio::Transformation.alignFace(p1)
|
1312
|
+
ft_p1 = flatZ( (ft.inverse * p1) )
|
1313
|
+
return false if ft_p1.empty?
|
1314
|
+
cw = OpenStudio.pointInPolygon(ft_p1.first, ft_p1, TOL)
|
1315
|
+
ft_p1 = flatZ( (ft.inverse * p1).reverse ) unless cw
|
1316
|
+
ft_p2 = flatZ( (ft.inverse * p2).reverse ) unless cw
|
1317
|
+
ft_p2 = flatZ( (ft.inverse * p2) ) if cw
|
1318
|
+
return false if ft_p2.empty?
|
1319
|
+
area1 = OpenStudio.getArea(ft_p1)
|
1320
|
+
area2 = OpenStudio.getArea(ft_p2)
|
1321
|
+
return empty("#{i1} area", mth, ERR, a) if area1.empty?
|
1322
|
+
return empty("#{i2} area", mth, ERR, a) if area2.empty?
|
1323
|
+
area1 = area1.get
|
1324
|
+
area2 = area2.get
|
1325
|
+
union = OpenStudio.join(ft_p1, ft_p2, TOL2)
|
1326
|
+
return false if union.empty?
|
1327
|
+
union = union.get
|
1328
|
+
area = OpenStudio.getArea(union)
|
1329
|
+
return empty("#{i1}:#{i2} union area", mth, ERR, a) if area.empty?
|
1330
|
+
area = area.get
|
1331
|
+
return false if area < TOL
|
1332
|
+
return true if (area - area2).abs < TOL
|
1333
|
+
return false if (area - area2).abs > TOL
|
1334
|
+
|
1335
|
+
true
|
1336
|
+
end
|
1337
|
+
|
1338
|
+
##
|
1339
|
+
# Validate whether an OpenStudio polygon overlaps another.
|
1340
|
+
#
|
1341
|
+
# @param p1 [OpenStudio::Point3dVector] or Point3D array of polygon #1
|
1342
|
+
# @param p2 [OpenStudio::Point3dVector] or Point3D array of polygon #2
|
1343
|
+
# @param id1 [String] polygon #1 identifier (optional)
|
1344
|
+
# @param id2 [String] polygon #2 identifier (optional)
|
1345
|
+
#
|
1346
|
+
# @return Returns true if polygons overlaps (or either fits into the other)
|
1347
|
+
# @return [Bool] false if invalid input
|
1348
|
+
def overlaps?(p1 = nil, p2 = nil, id1 = "", id2 = "")
|
1349
|
+
mth = "OSut::#{__callee__}"
|
1350
|
+
cl1 = OpenStudio::Point3dVector
|
1351
|
+
cl2 = OpenStudio::Point3d
|
1352
|
+
a = false
|
1353
|
+
|
1354
|
+
return invalid("id1", mth, 3, DBG, a) unless id1.respond_to?(:to_s)
|
1355
|
+
return invalid("id2", mth, 4, DBG, a) unless id2.respond_to?(:to_s)
|
1356
|
+
|
1357
|
+
i1 = id1.to_s
|
1358
|
+
i2 = id2.to_s
|
1359
|
+
i1 = "poly1" if i1.empty?
|
1360
|
+
i2 = "poly2" if i2.empty?
|
1361
|
+
|
1362
|
+
valid1 = p1.is_a?(cl1) || p1.is_a?(Array)
|
1363
|
+
valid2 = p2.is_a?(cl1) || p2.is_a?(Array)
|
1364
|
+
|
1365
|
+
return mismatch(i1, p1, cl1, mth, DBG, a) unless valid1
|
1366
|
+
return mismatch(i2, p2, cl1, mth, DBG, a) unless valid2
|
1367
|
+
return empty(i1, mth, ERR, a) if p1.empty?
|
1368
|
+
return empty(i2, mth, ERR, a) if p2.empty?
|
1369
|
+
|
1370
|
+
p1.each { |v| return mismatch(i1, v, cl2, mth, ERR, a) unless v.is_a?(cl2) }
|
1371
|
+
p2.each { |v| return mismatch(i2, v, cl2, mth, ERR, a) unless v.is_a?(cl2) }
|
1372
|
+
|
1373
|
+
# XY-plane transformation matrix ... needs to be clockwise for boost.
|
1374
|
+
ft = OpenStudio::Transformation.alignFace(p1)
|
1375
|
+
ft_p1 = flatZ( (ft.inverse * p1) )
|
1376
|
+
ft_p2 = flatZ( (ft.inverse * p2) )
|
1377
|
+
return false if ft_p1.empty?
|
1378
|
+
return false if ft_p2.empty?
|
1379
|
+
cw = OpenStudio.pointInPolygon(ft_p1.first, ft_p1, TOL)
|
1380
|
+
ft_p1 = flatZ( (ft.inverse * p1).reverse ) unless cw
|
1381
|
+
ft_p2 = flatZ( (ft.inverse * p2).reverse ) unless cw
|
1382
|
+
return false if ft_p1.empty?
|
1383
|
+
return false if ft_p2.empty?
|
1384
|
+
area1 = OpenStudio.getArea(ft_p1)
|
1385
|
+
area2 = OpenStudio.getArea(ft_p2)
|
1386
|
+
return empty("#{i1} area", mth, ERR, a) if area1.empty?
|
1387
|
+
return empty("#{i2} area", mth, ERR, a) if area2.empty?
|
1388
|
+
area1 = area1.get
|
1389
|
+
area2 = area2.get
|
1390
|
+
union = OpenStudio.join(ft_p1, ft_p2, TOL2)
|
1391
|
+
return false if union.empty?
|
1392
|
+
union = union.get
|
1393
|
+
area = OpenStudio.getArea(union)
|
1394
|
+
return empty("#{i1}:#{i2} union area", mth, ERR, a) if area.empty?
|
1395
|
+
area = area.get
|
1396
|
+
return false if area < TOL
|
1397
|
+
|
1398
|
+
true
|
1399
|
+
end
|
1400
|
+
|
1401
|
+
##
|
1402
|
+
# Generate offset vertices (by width) for a 3- or 4-sided, convex polygon.
|
1403
|
+
#
|
1404
|
+
# @param p1 [OpenStudio::Point3dVector] OpenStudio Point3D vector/array
|
1405
|
+
# @param w [Float] offset width (min: 0.0254m)
|
1406
|
+
# @param v [Integer] OpenStudio SDK version, eg '321' for 'v3.2.1' (optional)
|
1407
|
+
#
|
1408
|
+
# @return [OpenStudio::Point3dVector] offset points if successful
|
1409
|
+
# @return [OpenStudio::Point3dVector] original points if invalid input
|
1410
|
+
def offset(p1 = [], w = 0, v = 0)
|
1411
|
+
mth = "TBD::#{__callee__}"
|
1412
|
+
cl = OpenStudio::Point3d
|
1413
|
+
vrsn = OpenStudio.openStudioVersion.split(".").map(&:to_i).join.to_i
|
1414
|
+
|
1415
|
+
valid = p1.is_a?(OpenStudio::Point3dVector) || p1.is_a?(Array)
|
1416
|
+
return mismatch("pts", p1, cl1, mth, DBG, p1) unless valid
|
1417
|
+
return empty("pts", mth, ERR, p1) if p1.empty?
|
1418
|
+
valid = p1.size == 3 || p1.size == 4
|
1419
|
+
iv = true if p1.size == 4
|
1420
|
+
return invalid("pts", mth, 1, DBG, p1) unless valid
|
1421
|
+
return invalid("width", mth, 2, DBG, p1) unless w.respond_to?(:to_f)
|
1422
|
+
w = w.to_f
|
1423
|
+
return p1 if w < 0.0254
|
1424
|
+
v = v.to_i if v.respond_to?(:to_i)
|
1425
|
+
v = 0 unless v.respond_to?(:to_i)
|
1426
|
+
v = vrsn if v.zero?
|
1427
|
+
|
1428
|
+
p1.each { |x| return mismatch("p", x, cl, mth, ERR, p1) unless x.is_a?(cl) }
|
1429
|
+
|
1430
|
+
unless v < 340
|
1431
|
+
# XY-plane transformation matrix ... needs to be clockwise for boost.
|
1432
|
+
ft = OpenStudio::Transformation::alignFace(p1)
|
1433
|
+
ft_pts = flatZ( (ft.inverse * p1) )
|
1434
|
+
return p1 if ft_pts.empty?
|
1435
|
+
cw = OpenStudio::pointInPolygon(ft_pts.first, ft_pts, TOL)
|
1436
|
+
ft_pts = flatZ( (ft.inverse * p1).reverse ) unless cw
|
1437
|
+
offset = OpenStudio.buffer(ft_pts, w, TOL)
|
1438
|
+
return p1 if offset.empty?
|
1439
|
+
offset = offset.get
|
1440
|
+
offset = ft * offset if cw
|
1441
|
+
offset = (ft * offset).reverse unless cw
|
1442
|
+
|
1443
|
+
pz = OpenStudio::Point3dVector.new
|
1444
|
+
offset.each { |o| pz << OpenStudio::Point3d.new(o.x, o.y, o.z ) }
|
1445
|
+
return pz
|
1446
|
+
else # brute force approach
|
1447
|
+
pz = {}
|
1448
|
+
pz[:A] = {}
|
1449
|
+
pz[:B] = {}
|
1450
|
+
pz[:C] = {}
|
1451
|
+
pz[:D] = {} if iv
|
1452
|
+
|
1453
|
+
pz[:A][:p] = OpenStudio::Point3d.new(p1[0].x, p1[0].y, p1[0].z)
|
1454
|
+
pz[:B][:p] = OpenStudio::Point3d.new(p1[1].x, p1[1].y, p1[1].z)
|
1455
|
+
pz[:C][:p] = OpenStudio::Point3d.new(p1[2].x, p1[2].y, p1[2].z)
|
1456
|
+
pz[:D][:p] = OpenStudio::Point3d.new(p1[3].x, p1[3].y, p1[3].z) if iv
|
1457
|
+
|
1458
|
+
pzAp = pz[:A][:p]
|
1459
|
+
pzBp = pz[:B][:p]
|
1460
|
+
pzCp = pz[:C][:p]
|
1461
|
+
pzDp = pz[:D][:p] if iv
|
1462
|
+
|
1463
|
+
# Generate vector pairs, from next point & from previous point.
|
1464
|
+
# :f_n : "from next"
|
1465
|
+
# :f_p : "from previous"
|
1466
|
+
#
|
1467
|
+
#
|
1468
|
+
#
|
1469
|
+
#
|
1470
|
+
#
|
1471
|
+
#
|
1472
|
+
# A <---------- B
|
1473
|
+
# ^
|
1474
|
+
# \
|
1475
|
+
# \
|
1476
|
+
# C (or D)
|
1477
|
+
#
|
1478
|
+
pz[:A][:f_n] = pzAp - pzBp
|
1479
|
+
pz[:A][:f_p] = pzAp - pzCp unless iv
|
1480
|
+
pz[:A][:f_p] = pzAp - pzDp if iv
|
1481
|
+
|
1482
|
+
pz[:B][:f_n] = pzBp - pzCp
|
1483
|
+
pz[:B][:f_p] = pzBp - pzAp
|
1484
|
+
|
1485
|
+
pz[:C][:f_n] = pzCp - pzAp unless iv
|
1486
|
+
pz[:C][:f_n] = pzCp - pzDp if iv
|
1487
|
+
pz[:C][:f_p] = pzCp - pzBp
|
1488
|
+
|
1489
|
+
pz[:D][:f_n] = pzDp - pzAp if iv
|
1490
|
+
pz[:D][:f_p] = pzDp - pzCp if iv
|
1491
|
+
|
1492
|
+
# Generate 3D plane from vectors.
|
1493
|
+
#
|
1494
|
+
#
|
1495
|
+
# | <<< 3D plane ... from point A, with normal B>A
|
1496
|
+
# |
|
1497
|
+
# |
|
1498
|
+
# |
|
1499
|
+
# <---------- A <---------- B
|
1500
|
+
# |\
|
1501
|
+
# | \
|
1502
|
+
# | \
|
1503
|
+
# | C (or D)
|
1504
|
+
#
|
1505
|
+
pz[:A][:pl_f_n] = OpenStudio::Plane.new(pzAp, pz[:A][:f_n])
|
1506
|
+
pz[:A][:pl_f_p] = OpenStudio::Plane.new(pzAp, pz[:A][:f_p])
|
1507
|
+
|
1508
|
+
pz[:B][:pl_f_n] = OpenStudio::Plane.new(pzBp, pz[:B][:f_n])
|
1509
|
+
pz[:B][:pl_f_p] = OpenStudio::Plane.new(pzBp, pz[:B][:f_p])
|
1510
|
+
|
1511
|
+
pz[:C][:pl_f_n] = OpenStudio::Plane.new(pzCp, pz[:C][:f_n])
|
1512
|
+
pz[:C][:pl_f_p] = OpenStudio::Plane.new(pzCp, pz[:C][:f_p])
|
1513
|
+
|
1514
|
+
pz[:D][:pl_f_n] = OpenStudio::Plane.new(pzDp, pz[:D][:f_n]) if iv
|
1515
|
+
pz[:D][:pl_f_p] = OpenStudio::Plane.new(pzDp, pz[:D][:f_p]) if iv
|
1516
|
+
|
1517
|
+
# Project an extended point (pC) unto 3D plane.
|
1518
|
+
#
|
1519
|
+
# pC <<< projected unto extended B>A 3D plane
|
1520
|
+
# eC |
|
1521
|
+
# \ |
|
1522
|
+
# \ |
|
1523
|
+
# \|
|
1524
|
+
# <---------- A <---------- B
|
1525
|
+
# |\
|
1526
|
+
# | \
|
1527
|
+
# | \
|
1528
|
+
# | C (or D)
|
1529
|
+
#
|
1530
|
+
pz[:A][:p_n_pl] = pz[:A][:pl_f_n].project(pz[:A][:p] + pz[:A][:f_p])
|
1531
|
+
pz[:A][:n_p_pl] = pz[:A][:pl_f_p].project(pz[:A][:p] + pz[:A][:f_n])
|
1532
|
+
|
1533
|
+
pz[:B][:p_n_pl] = pz[:B][:pl_f_n].project(pz[:B][:p] + pz[:B][:f_p])
|
1534
|
+
pz[:B][:n_p_pl] = pz[:B][:pl_f_p].project(pz[:B][:p] + pz[:B][:f_n])
|
1535
|
+
|
1536
|
+
pz[:C][:p_n_pl] = pz[:C][:pl_f_n].project(pz[:C][:p] + pz[:C][:f_p])
|
1537
|
+
pz[:C][:n_p_pl] = pz[:C][:pl_f_p].project(pz[:C][:p] + pz[:C][:f_n])
|
1538
|
+
|
1539
|
+
pz[:D][:p_n_pl] = pz[:D][:pl_f_n].project(pz[:D][:p] + pz[:D][:f_p]) if iv
|
1540
|
+
pz[:D][:n_p_pl] = pz[:D][:pl_f_p].project(pz[:D][:p] + pz[:D][:f_n]) if iv
|
1541
|
+
|
1542
|
+
# Generate vector from point (e.g. A) to projected extended point (pC).
|
1543
|
+
#
|
1544
|
+
# pC
|
1545
|
+
# eC ^
|
1546
|
+
# \ |
|
1547
|
+
# \ |
|
1548
|
+
# \|
|
1549
|
+
# <---------- A <---------- B
|
1550
|
+
# |\
|
1551
|
+
# | \
|
1552
|
+
# | \
|
1553
|
+
# | C (or D)
|
1554
|
+
#
|
1555
|
+
pz[:A][:n_p_n_pl] = pz[:A][:p_n_pl] - pzAp
|
1556
|
+
pz[:A][:n_n_p_pl] = pz[:A][:n_p_pl] - pzAp
|
1557
|
+
|
1558
|
+
pz[:B][:n_p_n_pl] = pz[:B][:p_n_pl] - pzBp
|
1559
|
+
pz[:B][:n_n_p_pl] = pz[:B][:n_p_pl] - pzBp
|
1560
|
+
|
1561
|
+
pz[:C][:n_p_n_pl] = pz[:C][:p_n_pl] - pzCp
|
1562
|
+
pz[:C][:n_n_p_pl] = pz[:C][:n_p_pl] - pzCp
|
1563
|
+
|
1564
|
+
pz[:D][:n_p_n_pl] = pz[:D][:p_n_pl] - pzDp if iv
|
1565
|
+
pz[:D][:n_n_p_pl] = pz[:D][:n_p_pl] - pzDp if iv
|
1566
|
+
|
1567
|
+
# Fetch angle between both extended vectors (A>pC & A>pB),
|
1568
|
+
# ... then normalize (Cn).
|
1569
|
+
#
|
1570
|
+
# pC
|
1571
|
+
# eC ^
|
1572
|
+
# \ |
|
1573
|
+
# \ Cn
|
1574
|
+
# \|
|
1575
|
+
# <---------- A <---------- B
|
1576
|
+
# |\
|
1577
|
+
# | \
|
1578
|
+
# | \
|
1579
|
+
# | C (or D)
|
1580
|
+
#
|
1581
|
+
a1 = OpenStudio.getAngle(pz[:A][:n_p_n_pl], pz[:A][:n_n_p_pl])
|
1582
|
+
a2 = OpenStudio.getAngle(pz[:B][:n_p_n_pl], pz[:B][:n_n_p_pl])
|
1583
|
+
a3 = OpenStudio.getAngle(pz[:C][:n_p_n_pl], pz[:C][:n_n_p_pl])
|
1584
|
+
a4 = OpenStudio.getAngle(pz[:D][:n_p_n_pl], pz[:D][:n_n_p_pl]) if iv
|
1585
|
+
|
1586
|
+
# Generate new 3D points A', B', C' (and D') ... zigzag.
|
1587
|
+
#
|
1588
|
+
#
|
1589
|
+
#
|
1590
|
+
#
|
1591
|
+
# A' ---------------------- B'
|
1592
|
+
# \
|
1593
|
+
# \ A <---------- B
|
1594
|
+
# \ \
|
1595
|
+
# \ \
|
1596
|
+
# \ \
|
1597
|
+
# C' C
|
1598
|
+
pz[:A][:f_n].normalize
|
1599
|
+
pz[:A][:n_p_n_pl].normalize
|
1600
|
+
pzAp = pzAp + scalar(pz[:A][:n_p_n_pl], w)
|
1601
|
+
pzAp = pzAp + scalar(pz[:A][:f_n], w * Math.tan(a1/2))
|
1602
|
+
|
1603
|
+
pz[:B][:f_n].normalize
|
1604
|
+
pz[:B][:n_p_n_pl].normalize
|
1605
|
+
pzBp = pzBp + scalar(pz[:B][:n_p_n_pl], w)
|
1606
|
+
pzBp = pzBp + scalar(pz[:B][:f_n], w * Math.tan(a2/2))
|
1607
|
+
|
1608
|
+
pz[:C][:f_n].normalize
|
1609
|
+
pz[:C][:n_p_n_pl].normalize
|
1610
|
+
pzCp = pzCp + scalar(pz[:C][:n_p_n_pl], w)
|
1611
|
+
pzCp = pzCp + scalar(pz[:C][:f_n], w * Math.tan(a3/2))
|
1612
|
+
|
1613
|
+
pz[:D][:f_n].normalize if iv
|
1614
|
+
pz[:D][:n_p_n_pl].normalize if iv
|
1615
|
+
pzDp = pzDp + scalar(pz[:D][:n_p_n_pl], w) if iv
|
1616
|
+
pzDp = pzDp + scalar(pz[:D][:f_n], w * Math.tan(a4/2)) if iv
|
1617
|
+
|
1618
|
+
# Re-convert to OpenStudio 3D points.
|
1619
|
+
vec = OpenStudio::Point3dVector.new
|
1620
|
+
vec << OpenStudio::Point3d.new(pzAp.x, pzAp.y, pzAp.z)
|
1621
|
+
vec << OpenStudio::Point3d.new(pzBp.x, pzBp.y, pzBp.z)
|
1622
|
+
vec << OpenStudio::Point3d.new(pzCp.x, pzCp.y, pzCp.z)
|
1623
|
+
vec << OpenStudio::Point3d.new(pzDp.x, pzDp.y, pzDp.z) if iv
|
1624
|
+
|
1625
|
+
return vec
|
1626
|
+
end
|
1627
|
+
end
|
1628
|
+
|
1629
|
+
##
|
1630
|
+
# Callback when other modules extend OSlg
|
1631
|
+
#
|
1632
|
+
# @param base [Object] instance or class object
|
1633
|
+
def self.extended(base)
|
1634
|
+
base.send(:include, self)
|
1635
|
+
end
|
1636
|
+
end
|