honeybee-openstudio 1.8.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (71) hide show
  1. checksums.yaml +7 -0
  2. data/.coveralls.yml +1 -0
  3. data/.gitignore +32 -0
  4. data/.releaserc.json +7 -0
  5. data/.travis.yml +41 -0
  6. data/Gemfile +18 -0
  7. data/LICENSE.md +23 -0
  8. data/README.md +95 -0
  9. data/Rakefile +21 -0
  10. data/doc_templates/LICENSE.md +23 -0
  11. data/doc_templates/copyright_erb.txt +32 -0
  12. data/doc_templates/copyright_js.txt +5 -0
  13. data/doc_templates/copyright_ruby.txt +30 -0
  14. data/honeybee-openstudio.gemspec +35 -0
  15. data/lib/files/Honeybee.rb +113 -0
  16. data/lib/files/honeybee_workflow.osw +48 -0
  17. data/lib/files/urbanopt_Gemfile +32 -0
  18. data/lib/from_honeybee.rb +86 -0
  19. data/lib/from_honeybee/_openapi/model.json +8127 -0
  20. data/lib/from_honeybee/_openapi/simulation-parameter.json +843 -0
  21. data/lib/from_honeybee/construction/air.rb +65 -0
  22. data/lib/from_honeybee/construction/opaque.rb +77 -0
  23. data/lib/from_honeybee/construction/shade.rb +108 -0
  24. data/lib/from_honeybee/construction/window.rb +80 -0
  25. data/lib/from_honeybee/construction_set.rb +278 -0
  26. data/lib/from_honeybee/extension.rb +109 -0
  27. data/lib/from_honeybee/geometry/aperture.rb +167 -0
  28. data/lib/from_honeybee/geometry/door.rb +160 -0
  29. data/lib/from_honeybee/geometry/face.rb +163 -0
  30. data/lib/from_honeybee/geometry/room.rb +392 -0
  31. data/lib/from_honeybee/geometry/shade.rb +89 -0
  32. data/lib/from_honeybee/hvac/ideal_air.rb +150 -0
  33. data/lib/from_honeybee/load/electric_equipment.rb +96 -0
  34. data/lib/from_honeybee/load/gas_equipment.rb +97 -0
  35. data/lib/from_honeybee/load/infiltration.rb +95 -0
  36. data/lib/from_honeybee/load/lighting.rb +98 -0
  37. data/lib/from_honeybee/load/people.rb +100 -0
  38. data/lib/from_honeybee/load/setpoint_humidistat.rb +75 -0
  39. data/lib/from_honeybee/load/setpoint_thermostat.rb +71 -0
  40. data/lib/from_honeybee/load/ventilation.rb +96 -0
  41. data/lib/from_honeybee/material/opaque.rb +94 -0
  42. data/lib/from_honeybee/material/opaque_no_mass.rb +94 -0
  43. data/lib/from_honeybee/material/window_blind.rb +238 -0
  44. data/lib/from_honeybee/material/window_gas.rb +76 -0
  45. data/lib/from_honeybee/material/window_gas_custom.rb +118 -0
  46. data/lib/from_honeybee/material/window_gas_mixture.rb +79 -0
  47. data/lib/from_honeybee/material/window_glazing.rb +166 -0
  48. data/lib/from_honeybee/material/window_shade.rb +160 -0
  49. data/lib/from_honeybee/material/window_simpleglazsys.rb +73 -0
  50. data/lib/from_honeybee/model.rb +434 -0
  51. data/lib/from_honeybee/model_object.rb +110 -0
  52. data/lib/from_honeybee/program_type.rb +124 -0
  53. data/lib/from_honeybee/schedule/fixed_interval.rb +115 -0
  54. data/lib/from_honeybee/schedule/ruleset.rb +164 -0
  55. data/lib/from_honeybee/schedule/type_limit.rb +88 -0
  56. data/lib/from_honeybee/simulation/designday.rb +105 -0
  57. data/lib/from_honeybee/simulation/extension.rb +46 -0
  58. data/lib/from_honeybee/simulation/parameter.rb +277 -0
  59. data/lib/from_honeybee/version.rb +34 -0
  60. data/lib/measures/.gitkeep +0 -0
  61. data/lib/measures/from_honeybee_model/LICENSE.md +27 -0
  62. data/lib/measures/from_honeybee_model/README.md +32 -0
  63. data/lib/measures/from_honeybee_model/measure.rb +91 -0
  64. data/lib/measures/from_honeybee_model/measure.xml +103 -0
  65. data/lib/measures/from_honeybee_model/tests/from_honeybee_model_test.rb +126 -0
  66. data/lib/measures/from_honeybee_simulation_parameter/LICENSE.md +27 -0
  67. data/lib/measures/from_honeybee_simulation_parameter/README.md +32 -0
  68. data/lib/measures/from_honeybee_simulation_parameter/measure.rb +95 -0
  69. data/lib/measures/from_honeybee_simulation_parameter/measure.xml +91 -0
  70. data/lib/measures/from_honeybee_simulation_parameter/tests/from_honeybee_simulation_parameter_test.rb +109 -0
  71. metadata +243 -0
@@ -0,0 +1,109 @@
1
+ # *******************************************************************************
2
+ # Honeybee OpenStudio Gem, Copyright (c) 2020, Alliance for Sustainable
3
+ # Energy, LLC, Ladybug Tools LLC and other contributors. All rights reserved.
4
+ #
5
+ # Redistribution and use in source and binary forms, with or without
6
+ # modification, are permitted provided that the following conditions are met:
7
+ #
8
+ # (1) Redistributions of source code must retain the above copyright notice,
9
+ # this list of conditions and the following disclaimer.
10
+ #
11
+ # (2) Redistributions in binary form must reproduce the above copyright notice,
12
+ # this list of conditions and the following disclaimer in the documentation
13
+ # and/or other materials provided with the distribution.
14
+ #
15
+ # (3) Neither the name of the copyright holder nor the names of any contributors
16
+ # may be used to endorse or promote products derived from this software without
17
+ # specific prior written permission from the respective party.
18
+ #
19
+ # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS
20
+ # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
21
+ # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22
+ # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE
23
+ # UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF
24
+ # THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
25
+ # EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
26
+ # OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27
+ # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
28
+ # STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29
+ # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
+ # *******************************************************************************
31
+
32
+ require 'openstudio/extension'
33
+
34
+ # NOTE: This file has been derived from one within the openStudio-extension gem
35
+ # The properties here are a standard part of openstudio extensions
36
+
37
+ module FromHoneybee
38
+ class Extension < OpenStudio::Extension::Extension
39
+ @@schema = nil
40
+
41
+ # Override parent class
42
+ def initialize
43
+ super
44
+
45
+ # Note that the root_dir is only meaningful when the gem is in a github repository
46
+ # When installed as a Ruby gem, the highest directory within the gem is the lib_dir
47
+ @root_dir = File.absolute_path(File.join(File.dirname(__FILE__), '..', '..'))
48
+ @lib_dir = File.absolute_path(File.join(File.dirname(__FILE__), '..'))
49
+
50
+ @instance_lock = Mutex.new
51
+ @@schema ||= schema
52
+ end
53
+
54
+ # Return the absolute path of the measures or nil if there is none.
55
+ # Can be used when configuring OSWs
56
+ def measures_dir
57
+ File.absolute_path(File.join(@lib_dir, 'measures'))
58
+ end
59
+
60
+ # Relevant files such as the openapi JSON schema files for the honeybee model.
61
+ # Return the absolute path of the files or nil if there is none.
62
+ # Used when configuring OSWs
63
+ def files_dir
64
+ File.absolute_path(File.join(@lib_dir, 'files'))
65
+ end
66
+
67
+ # Doc templates are common files like copyright files which are used to update measures
68
+ # Doc templates will only be applied when the gem is a part of a repository
69
+ # Return the absolute path of the doc templates dir or nil if there is none
70
+ def doc_templates_dir
71
+ File.absolute_path(File.join(@root_dir, 'doc_templates'))
72
+ end
73
+
74
+ # return path to the model schema file
75
+ def schema_file
76
+ File.join(@lib_dir, 'from_honeybee', '_openapi', 'model.json')
77
+ end
78
+
79
+ # return the model schema
80
+ def schema
81
+ @instance_lock.synchronize do
82
+ if @@schema.nil?
83
+ File.open(schema_file, 'r') do |f|
84
+ @@schema = JSON.parse(f.read, symbolize_names: true)
85
+ end
86
+ end
87
+ end
88
+
89
+ @@schema
90
+ end
91
+
92
+ # check if the model schema is valid
93
+ def schema_valid?
94
+ if Gem.loaded_specs.has_key?("json-schema")
95
+ require 'json-schema'
96
+ metaschema = JSON::Validator.validator_for_name('draft6').metaschema
97
+ JSON::Validator.validate(metaschema, @@schema)
98
+ end
99
+ end
100
+
101
+ # return detailed schema validation errors
102
+ def schema_validation_errors
103
+ if Gem.loaded_specs.has_key?("json-schema")
104
+ metaschema = JSON::Validator.validator_for_name('draft6').metaschema
105
+ JSON::Validator.fully_validate(metaschema, @@schema)
106
+ end
107
+ end
108
+ end
109
+ end
@@ -0,0 +1,167 @@
1
+ # *******************************************************************************
2
+ # Honeybee OpenStudio Gem, Copyright (c) 2020, Alliance for Sustainable
3
+ # Energy, LLC, Ladybug Tools LLC and other contributors. All rights reserved.
4
+ #
5
+ # Redistribution and use in source and binary forms, with or without
6
+ # modification, are permitted provided that the following conditions are met:
7
+ #
8
+ # (1) Redistributions of source code must retain the above copyright notice,
9
+ # this list of conditions and the following disclaimer.
10
+ #
11
+ # (2) Redistributions in binary form must reproduce the above copyright notice,
12
+ # this list of conditions and the following disclaimer in the documentation
13
+ # and/or other materials provided with the distribution.
14
+ #
15
+ # (3) Neither the name of the copyright holder nor the names of any contributors
16
+ # may be used to endorse or promote products derived from this software without
17
+ # specific prior written permission from the respective party.
18
+ #
19
+ # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS
20
+ # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
21
+ # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22
+ # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE
23
+ # UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF
24
+ # THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
25
+ # EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
26
+ # OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27
+ # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
28
+ # STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29
+ # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
+ # *******************************************************************************
31
+
32
+ require 'from_honeybee/model_object'
33
+
34
+ require 'openstudio'
35
+
36
+ module FromHoneybee
37
+ class Aperture < ModelObject
38
+ attr_reader :errors, :warnings
39
+
40
+ def initialize(hash)
41
+ super(hash)
42
+ raise "Incorrect model type '#{@type}'" unless @type == 'Aperture'
43
+ end
44
+
45
+ def defaults
46
+ @@schema[:components][:schemas][:ApertureEnergyPropertiesAbridged][:properties]
47
+ end
48
+
49
+ def find_existing_openstudio_object(openstudio_model)
50
+ object = openstudio_model.getSubSurfaceByName(@hash[:identifier])
51
+ return object.get if object.is_initialized
52
+ nil
53
+ end
54
+
55
+ def to_openstudio(openstudio_model)
56
+ # create the OpenStudio aperture object
57
+ os_vertices = OpenStudio::Point3dVector.new
58
+ @hash[:geometry][:boundary].each do |vertex|
59
+ os_vertices << OpenStudio::Point3d.new(vertex[0], vertex[1], vertex[2])
60
+ end
61
+ reordered_vertices = OpenStudio.reorderULC(os_vertices)
62
+
63
+ # triangulate subsurface if neccesary
64
+ triangulated = false
65
+ final_vertices_list = []
66
+ matching_os_subsurfaces = []
67
+ matching_os_subsurface_indices = []
68
+ if reordered_vertices.size > 4
69
+
70
+ # if this apeture has a matched apeture, see if the other one has already been created
71
+ # the matched apeture should have been converted to multiple subsurfaces
72
+ if @hash[:boundary_condition][:type] == 'Surface'
73
+ adj_srf_identifier = @hash[:boundary_condition][:boundary_condition_objects][0]
74
+ regex = Regexp.new("#{adj_srf_identifier}\.\.(\\d+)")
75
+ openstudio_model.getSubSurfaces.each do |subsurface|
76
+ if md = regex.match(subsurface.nameString)
77
+ final_vertices_list << OpenStudio.reorderULC(OpenStudio::reverse(subsurface.vertices))
78
+ matching_os_subsurfaces << subsurface
79
+ matching_os_subsurface_indices << md[1]
80
+ end
81
+ end
82
+ end
83
+
84
+ # if other apeture is not already created, do the triangulation
85
+ if final_vertices_list.empty?
86
+
87
+ # transform to face coordinates
88
+ t = OpenStudio::Transformation::alignFace(reordered_vertices)
89
+ tInv = t.inverse
90
+ face_vertices = OpenStudio::reverse(tInv*reordered_vertices)
91
+
92
+ # no holes in the subsurface
93
+ holes = OpenStudio::Point3dVectorVector.new
94
+
95
+ # triangulate surface
96
+ triangles = OpenStudio::computeTriangulation(face_vertices, holes)
97
+ if triangles.empty?
98
+ raise "Failed to triangulate aperture #{@hash[:identifier]} with #{reordered_vertices.size} vertices"
99
+ end
100
+
101
+ # create new list of surfaces
102
+ triangles.each do |vertices|
103
+ final_vertices_list << OpenStudio.reorderULC(OpenStudio::reverse(t*vertices))
104
+ end
105
+
106
+ triangulated = true
107
+
108
+ end
109
+
110
+ else
111
+ # reordered_vertices are good as is
112
+ final_vertices_list << reordered_vertices
113
+ end
114
+
115
+ result = []
116
+ final_vertices_list.each_with_index do |reordered_vertices, index|
117
+ os_subsurface = OpenStudio::Model::SubSurface.new(reordered_vertices, openstudio_model)
118
+
119
+ if !matching_os_subsurfaces.empty?
120
+ os_subsurface.setName(@hash[:identifier] + "..#{matching_os_subsurface_indices[index]}")
121
+ elsif triangulated
122
+ os_subsurface.setName(@hash[:identifier] + "..#{index}")
123
+ else
124
+ os_subsurface.setName(@hash[:identifier])
125
+ end
126
+
127
+ # assign the construction if it exists
128
+ if @hash[:properties][:energy][:construction]
129
+ construction_identifier = @hash[:properties][:energy][:construction]
130
+ construction = openstudio_model.getConstructionByName(construction_identifier)
131
+ unless construction.empty?
132
+ os_construction = construction.get
133
+ os_subsurface.setConstruction(os_construction)
134
+ end
135
+ end
136
+
137
+ # assign the boundary condition object if it's a Surface
138
+ if @hash[:boundary_condition][:type] == 'Surface'
139
+ if !matching_os_subsurfaces.empty?
140
+ # we already have the match because this was created from the matching_os_subsurfaces
141
+ # setAdjacentSubSurface will fail at this point because sub surface is not assigned to surface yet, store data for later
142
+ adj_srf_identifier = matching_os_subsurfaces[index].nameString
143
+ os_subsurface.additionalProperties.setFeature("AdjacentSubSurfaceName", adj_srf_identifier)
144
+ elsif triangulated
145
+ # other subsurfaces haven't been created yet, no-op
146
+ else
147
+ # get adjacent sub surface by identifier from openstudio model
148
+ # setAdjacentSubSurface will fail at this point because sub surface is not assigned to surface yet, store data for later
149
+ adj_srf_identifier = @hash[:boundary_condition][:boundary_condition_objects][0]
150
+ os_subsurface.additionalProperties.setFeature("AdjacentSubSurfaceName", adj_srf_identifier)
151
+ end
152
+ end
153
+
154
+ # assign the operable property
155
+ if @hash[:is_operable] == false
156
+ os_subsurface.setSubSurfaceType('FixedWindow')
157
+ else
158
+ os_subsurface.setSubSurfaceType('OperableWindow')
159
+ end
160
+
161
+ result << os_subsurface
162
+ end
163
+
164
+ return result
165
+ end
166
+ end # Aperture
167
+ end # FromHoneybee
@@ -0,0 +1,160 @@
1
+ # *******************************************************************************
2
+ # Honeybee OpenStudio Gem, Copyright (c) 2020, Alliance for Sustainable
3
+ # Energy, LLC, Ladybug Tools LLC and other contributors. All rights reserved.
4
+ #
5
+ # Redistribution and use in source and binary forms, with or without
6
+ # modification, are permitted provided that the following conditions are met:
7
+ #
8
+ # (1) Redistributions of source code must retain the above copyright notice,
9
+ # this list of conditions and the following disclaimer.
10
+ #
11
+ # (2) Redistributions in binary form must reproduce the above copyright notice,
12
+ # this list of conditions and the following disclaimer in the documentation
13
+ # and/or other materials provided with the distribution.
14
+ #
15
+ # (3) Neither the name of the copyright holder nor the names of any contributors
16
+ # may be used to endorse or promote products derived from this software without
17
+ # specific prior written permission from the respective party.
18
+ #
19
+ # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS
20
+ # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
21
+ # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22
+ # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE
23
+ # UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF
24
+ # THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
25
+ # EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
26
+ # OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27
+ # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
28
+ # STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29
+ # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
+ # *******************************************************************************
31
+
32
+ require 'from_honeybee/model_object'
33
+
34
+ require 'openstudio'
35
+
36
+ module FromHoneybee
37
+ class Door < ModelObject
38
+ attr_reader :errors, :warnings
39
+
40
+ def initialize(hash)
41
+ super(hash)
42
+ raise "Incorrect model type '#{@type}'" unless @type == 'Door'
43
+ end
44
+
45
+ def defaults
46
+ @@schema[:components][:schemas][:DoorEnergyPropertiesAbridged][:properties]
47
+ end
48
+
49
+ def find_existing_openstudio_object(openstudio_model)
50
+ object = openstudio_model.getSubSurfaceByName(@hash[:identifier])
51
+ return object.get if object.is_initialized
52
+ nil
53
+ end
54
+
55
+ def to_openstudio(openstudio_model)
56
+ # create the OpenStudio door object
57
+ os_vertices = OpenStudio::Point3dVector.new
58
+ @hash[:geometry][:boundary].each do |vertex|
59
+ os_vertices << OpenStudio::Point3d.new(vertex[0], vertex[1], vertex[2])
60
+ end
61
+ reordered_vertices = OpenStudio.reorderULC(os_vertices)
62
+
63
+ # triangulate subsurface if neccesary
64
+ triangulated = false
65
+ final_vertices_list = []
66
+ matching_os_subsurfaces = []
67
+ matching_os_subsurface_indices = []
68
+ if reordered_vertices.size > 4
69
+
70
+ # if this door has a matched door, see if the other one has already been created
71
+ # the matched door should have been converted to multiple subsurfaces
72
+ if @hash[:boundary_condition][:type] == 'Surface'
73
+ adj_srf_identifier = @hash[:boundary_condition][:boundary_condition_objects][0]
74
+ regex = Regexp.new("#{adj_srf_identifier}\.\.(\\d+)")
75
+ openstudio_model.getSubSurfaces.each do |subsurface|
76
+ if md = regex.match(subsurface.nameString)
77
+ final_vertices_list << OpenStudio.reorderULC(OpenStudio::reverse(subsurface.vertices))
78
+ matching_os_subsurfaces << subsurface
79
+ matching_os_subsurface_indices << md[1]
80
+ end
81
+ end
82
+ end
83
+
84
+ # if other door is not already created, do the triangulation
85
+ if final_vertices_list.empty?
86
+
87
+ # transform to face coordinates
88
+ t = OpenStudio::Transformation::alignFace(reordered_vertices)
89
+ tInv = t.inverse
90
+ face_vertices = OpenStudio::reverse(tInv*reordered_vertices)
91
+
92
+ # no holes in the subsurface
93
+ holes = OpenStudio::Point3dVectorVector.new
94
+
95
+ # triangulate surface
96
+ triangles = OpenStudio::computeTriangulation(face_vertices, holes)
97
+ if triangles.empty?
98
+ raise "Failed to triangulate door #{@hash[:identifier]} with #{reordered_vertices.size} vertices"
99
+ end
100
+
101
+ # create new list of surfaces
102
+ triangles.each do |vertices|
103
+ final_vertices_list << OpenStudio.reorderULC(OpenStudio::reverse(t*vertices))
104
+ end
105
+
106
+ triangulated = true
107
+
108
+ end
109
+
110
+ else
111
+ # reordered_vertices are good as is
112
+ final_vertices_list << reordered_vertices
113
+ end
114
+
115
+ result = []
116
+ final_vertices_list.each_with_index do |reordered_vertices, index|
117
+ os_subsurface = OpenStudio::Model::SubSurface.new(reordered_vertices, openstudio_model)
118
+
119
+ if !matching_os_subsurfaces.empty?
120
+ os_subsurface.setName(@hash[:identifier] + "..#{matching_os_subsurface_indices[index]}")
121
+ elsif triangulated
122
+ os_subsurface.setName(@hash[:identifier] + "..#{index}")
123
+ else
124
+ os_subsurface.setName(@hash[:identifier])
125
+ end
126
+
127
+ # assign the construction if it exists
128
+ if @hash[:properties][:energy][:construction]
129
+ construction_identifier = @hash[:properties][:energy][:construction]
130
+ construction = openstudio_model.getConstructionByName(construction_identifier)
131
+ unless construction.empty?
132
+ os_construction = construction.get
133
+ os_subsurface.setConstruction(os_construction)
134
+ end
135
+ end
136
+
137
+ # assign the boundary condition object if it's a Surface
138
+ if @hash[:boundary_condition][:type] == 'Surface'
139
+ if !matching_os_subsurfaces.empty?
140
+ # we already have the match because this was created from the matching_os_subsurfaces
141
+ # setAdjacentSubSurface will fail at this point because sub surface is not assigned to surface yet, store data for later
142
+ adj_srf_identifier = matching_os_subsurfaces[index].nameString
143
+ os_subsurface.additionalProperties.setFeature("AdjacentSubSurfaceName", adj_srf_identifier)
144
+ elsif triangulated
145
+ # other subsurfaces haven't been created yet, no-op
146
+ else
147
+ # get adjacent sub surface by identifier from openstudio model
148
+ # setAdjacentSubSurface will fail at this point because sub surface is not assigned to surface yet, store data for later
149
+ adj_srf_identifier = @hash[:boundary_condition][:boundary_condition_objects][0]
150
+ os_subsurface.additionalProperties.setFeature("AdjacentSubSurfaceName", adj_srf_identifier)
151
+ end
152
+ end
153
+
154
+ result << os_subsurface
155
+ end
156
+
157
+ return result
158
+ end
159
+ end # Door
160
+ end # FromHoneybee
@@ -0,0 +1,163 @@
1
+ # *******************************************************************************
2
+ # Honeybee OpenStudio Gem, Copyright (c) 2020, Alliance for Sustainable
3
+ # Energy, LLC, Ladybug Tools LLC and other contributors. All rights reserved.
4
+ #
5
+ # Redistribution and use in source and binary forms, with or without
6
+ # modification, are permitted provided that the following conditions are met:
7
+ #
8
+ # (1) Redistributions of source code must retain the above copyright notice,
9
+ # this list of conditions and the following disclaimer.
10
+ #
11
+ # (2) Redistributions in binary form must reproduce the above copyright notice,
12
+ # this list of conditions and the following disclaimer in the documentation
13
+ # and/or other materials provided with the distribution.
14
+ #
15
+ # (3) Neither the name of the copyright holder nor the names of any contributors
16
+ # may be used to endorse or promote products derived from this software without
17
+ # specific prior written permission from the respective party.
18
+ #
19
+ # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) AND ANY CONTRIBUTORS
20
+ # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
21
+ # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22
+ # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S), ANY CONTRIBUTORS, THE
23
+ # UNITED STATES GOVERNMENT, OR THE UNITED STATES DEPARTMENT OF ENERGY, NOR ANY OF
24
+ # THEIR EMPLOYEES, BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
25
+ # EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
26
+ # OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27
+ # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
28
+ # STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29
+ # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
+ # *******************************************************************************
31
+
32
+ require 'from_honeybee/model_object'
33
+ require 'from_honeybee/geometry/aperture'
34
+ require 'from_honeybee/geometry/door'
35
+
36
+ require 'openstudio'
37
+
38
+ module FromHoneybee
39
+ class Face < ModelObject
40
+ attr_reader :errors, :warnings
41
+
42
+ def initialize(hash = {})
43
+ super(hash)
44
+
45
+ raise "Incorrect model type '#{@type}'" unless @type == 'Face'
46
+ end
47
+
48
+ def defaults
49
+ @@schema[:components][:schemas][:FaceEnergyPropertiesAbridged][:properties]
50
+ end
51
+
52
+ def find_existing_openstudio_object(openstudio_model)
53
+ model_surf = openstudio_model.getSurfaceByName(@hash[:identifier])
54
+ return model_surf.get unless model_surf.empty?
55
+ nil
56
+ end
57
+
58
+ def to_openstudio(openstudio_model)
59
+ # create the openstudio surface
60
+ os_vertices = OpenStudio::Point3dVector.new
61
+ @hash[:geometry][:boundary].each do |vertex|
62
+ os_vertices << OpenStudio::Point3d.new(vertex[0], vertex[1], vertex[2])
63
+ end
64
+ reordered_vertices = OpenStudio.reorderULC(os_vertices)
65
+
66
+ os_surface = OpenStudio::Model::Surface.new(reordered_vertices, openstudio_model)
67
+ os_surface.setName(@hash[:identifier])
68
+ os_surface.setSurfaceType(@hash[:face_type])
69
+ # assign the construction if it is present
70
+ if @hash[:properties][:energy][:construction]
71
+ construction_identifier = @hash[:properties][:energy][:construction]
72
+ construction = openstudio_model.getConstructionByName(construction_identifier)
73
+ unless construction.empty?
74
+ os_construction = construction.get
75
+ os_surface.setConstruction(os_construction)
76
+ end
77
+ end
78
+
79
+ # assign the boundary condition
80
+ boundary_condition = (@hash[:boundary_condition][:type])
81
+ case boundary_condition
82
+ when 'Outdoors'
83
+ if @hash[:boundary_condition][:sun_exposure] == false
84
+ os_surface.setSunExposure('NoSun')
85
+ else
86
+ os_surface.setSunExposure('SunExposed')
87
+ end
88
+ if @hash[:boundary_condition][:wind_exposure] == false
89
+ os_surface.setWindExposure('NoWind')
90
+ else
91
+ os_surface.setWindExposure('WindExposed')
92
+ end
93
+ if @hash[:boundary_condition][:view_factor].is_a? Numeric
94
+ os_surface.setViewFactortoGround(@hash[:boundary_condition][:view_factor])
95
+ else
96
+ os_surface.autocalculateViewFactortoGround
97
+ end
98
+ when 'Surface'
99
+ # get adjacent surface by identifier from openstudio model
100
+ adj_srf_identifier = @hash[:boundary_condition][:boundary_condition_objects][0]
101
+ surface_object = openstudio_model.getSurfaceByName(adj_srf_identifier)
102
+ unless surface_object.empty?
103
+ surface = surface_object.get
104
+ os_surface.setAdjacentSurface(surface)
105
+ end
106
+ end
107
+ unless @hash[:boundary_condition][:type] == 'Surface'
108
+ os_surface.setOutsideBoundaryCondition(@hash[:boundary_condition][:type])
109
+ end
110
+
111
+ # assign apertures if they exist
112
+ if @hash[:apertures]
113
+ @hash[:apertures].each do |aperture|
114
+ ladybug_aperture = Aperture.new(aperture)
115
+ os_subsurface_apertures = ladybug_aperture.to_openstudio(openstudio_model)
116
+ os_subsurface_apertures.each do |os_subsurface_aperture|
117
+ if @hash[:face_type] == 'RoofCeiling' or @hash[:face_type] == 'Floor'
118
+ if @hash[:boundary_condition][:type] == 'Outdoors' && aperture[:is_operable] == false
119
+ os_subsurface_aperture.setSubSurfaceType('Skylight')
120
+ end
121
+ end
122
+ os_subsurface_aperture.setSurface(os_surface)
123
+ end
124
+ end
125
+ end
126
+
127
+ # assign doors if they exist
128
+ if @hash[:doors]
129
+ @hash[:doors].each do |door|
130
+ honeybee_door = Door.new(door)
131
+ os_subsurface_doors = honeybee_door.to_openstudio(openstudio_model)
132
+ os_subsurface_doors.each do |os_subsurface_door|
133
+ os_subsurface_door.setSurface(os_surface)
134
+ if door[:is_glass] == true
135
+ os_subsurface_door.setSubSurfaceType('GlassDoor')
136
+ elsif (@hash[:face_type] == 'RoofCeiling' or @hash[:face_type] == 'Floor') && @hash[:boundary_condition][:type] == 'Outdoors'
137
+ os_subsurface_door.setSubSurfaceType('OverheadDoor')
138
+ elsif door[:is_glass] == false or door[:is_glass].nil?
139
+ os_subsurface_door.setSubSurfaceType('Door')
140
+ end
141
+ end
142
+ end
143
+ end
144
+
145
+ os_surface.subSurfaces.each do |os_subsurface|
146
+ if os_subsurface.hasAdditionalProperties
147
+ adj_sub_srf_identifier = os_subsurface.additionalProperties.getFeatureAsString("AdjacentSubSurfaceName")
148
+ unless adj_sub_srf_identifier.empty?
149
+ adj_sub_srf = openstudio_model.getSubSurfaceByName(adj_sub_srf_identifier.get)
150
+ unless adj_sub_srf.empty?
151
+ os_subsurface.setAdjacentSubSurface(adj_sub_srf.get)
152
+ end
153
+ end
154
+
155
+ # clean up, we don't need this object any more
156
+ os_subsurface.removeAdditionalProperties
157
+ end
158
+ end
159
+
160
+ os_surface
161
+ end
162
+ end # Face
163
+ end # FromHoneybee