Spaces:
Sleeping
Sleeping
Update app/component_selection.py
Browse files- app/component_selection.py +117 -138
app/component_selection.py
CHANGED
@@ -4,6 +4,7 @@ Provides UI for selecting building components in the HVAC Load Calculator.
|
|
4 |
All dependencies are included within this file for standalone operation.
|
5 |
Updated 2025-04-28: Added surface color selection, expanded component types, and skylight support.
|
6 |
Updated 2025-05-02: Added crack dimensions for walls/doors, drapery properties for windows/skylights, and roof height per enhancement plan.
|
|
|
7 |
|
8 |
Author: Dr Majed Abuseif
|
9 |
"""
|
@@ -39,11 +40,6 @@ class ComponentType(Enum):
|
|
39 |
DOOR = "Door"
|
40 |
SKYLIGHT = "Skylight"
|
41 |
|
42 |
-
class SurfaceColor(Enum):
|
43 |
-
DARK = "Dark"
|
44 |
-
MEDIUM = "Medium"
|
45 |
-
LIGHT = "Light"
|
46 |
-
|
47 |
class GlazingType(Enum):
|
48 |
SINGLE_CLEAR = "Single Clear"
|
49 |
SINGLE_TINTED = "Single Tinted"
|
@@ -104,7 +100,6 @@ class Wall(BuildingComponent):
|
|
104 |
absorptivity: float = 0.6
|
105 |
shading_coefficient: float = 1.0
|
106 |
infiltration_rate_cfm: float = 0.0
|
107 |
-
surface_color: str = "Dark" # Added surface color
|
108 |
crack_length: float = 0.0 # Added for infiltration (m)
|
109 |
crack_width: float = 0.0 # Added for infiltration (m)
|
110 |
|
@@ -125,17 +120,13 @@ class Wall(BuildingComponent):
|
|
125 |
if self.wall_group not in VALID_WALL_GROUPS:
|
126 |
st.warning(f"Invalid wall_group '{self.wall_group}' for wall '{self.name}'. Defaulting to 'A'.")
|
127 |
self.wall_group = "A"
|
128 |
-
VALID_SURFACE_COLORS = {"Dark", "Medium", "Light"}
|
129 |
-
if self.surface_color not in VALID_SURFACE_COLORS:
|
130 |
-
st.warning(f"Invalid surface_color '{self.surface_color}' for wall '{self.name}'. Defaulting to 'Dark'.")
|
131 |
-
self.surface_color = "Dark"
|
132 |
|
133 |
def to_dict(self) -> dict:
|
134 |
base_dict = super().to_dict()
|
135 |
base_dict.update({
|
136 |
"wall_type": self.wall_type, "wall_group": self.wall_group, "absorptivity": self.absorptivity,
|
137 |
"shading_coefficient": self.shading_coefficient, "infiltration_rate_cfm": self.infiltration_rate_cfm,
|
138 |
-
"
|
139 |
})
|
140 |
return base_dict
|
141 |
|
@@ -145,7 +136,6 @@ class Roof(BuildingComponent):
|
|
145 |
roof_group: str = "A" # ASHRAE group
|
146 |
slope: str = "Flat"
|
147 |
absorptivity: float = 0.6
|
148 |
-
surface_color: str = "Dark" # Added surface color
|
149 |
roof_height: float = 3.0 # Added for stack effect (m)
|
150 |
|
151 |
def __post_init__(self):
|
@@ -161,17 +151,12 @@ class Roof(BuildingComponent):
|
|
161 |
if self.roof_group not in VALID_ROOF_GROUPS:
|
162 |
st.warning(f"Invalid roof_group '{self.roof_group}' for roof '{self.name}'. Defaulting to 'A'.")
|
163 |
self.roof_group = "A"
|
164 |
-
VALID_SURFACE_COLORS = {"Dark", "Medium", "Light"}
|
165 |
-
if self.surface_color not in VALID_SURFACE_COLORS:
|
166 |
-
st.warning(f"Invalid surface_color '{self.surface_color}' for roof '{self.name}'. Defaulting to 'Dark'.")
|
167 |
-
self.surface_color = "Dark"
|
168 |
|
169 |
def to_dict(self) -> dict:
|
170 |
base_dict = super().to_dict()
|
171 |
base_dict.update({
|
172 |
"roof_type": self.roof_type, "roof_group": self.roof_group, "slope": self.slope,
|
173 |
-
"absorptivity": self.absorptivity, "
|
174 |
-
"roof_height": self.roof_height
|
175 |
})
|
176 |
return base_dict
|
177 |
|
@@ -206,13 +191,13 @@ class Window(BuildingComponent):
|
|
206 |
shgc: float = 0.7
|
207 |
shading_device: str = "None"
|
208 |
shading_coefficient: float = 1.0
|
209 |
-
frame_type: str = "Aluminum without Thermal Break"
|
210 |
frame_percentage: float = 20.0
|
211 |
infiltration_rate_cfm: float = 0.0
|
212 |
-
glazing_type: str = "Double Clear"
|
213 |
-
drapery_openness: str = "Open"
|
214 |
-
drapery_color: str = "Light"
|
215 |
-
drapery_fullness: float = 1.5
|
216 |
|
217 |
def __post_init__(self):
|
218 |
super().__post_init__()
|
@@ -251,8 +236,8 @@ class Window(BuildingComponent):
|
|
251 |
class Door(BuildingComponent):
|
252 |
door_type: str = "Solid Wood"
|
253 |
infiltration_rate_cfm: float = 0.0
|
254 |
-
crack_length: float = 0.0
|
255 |
-
crack_width: float = 0.0
|
256 |
|
257 |
def __post_init__(self):
|
258 |
super().__post_init__()
|
@@ -277,13 +262,13 @@ class Skylight(BuildingComponent):
|
|
277 |
shgc: float = 0.7
|
278 |
shading_device: str = "None"
|
279 |
shading_coefficient: float = 1.0
|
280 |
-
frame_type: str = "Aluminum without Thermal Break"
|
281 |
frame_percentage: float = 20.0
|
282 |
infiltration_rate_cfm: float = 0.0
|
283 |
-
glazing_type: str = "Double Clear"
|
284 |
-
drapery_openness: str = "Open"
|
285 |
-
drapery_color: str = "Light"
|
286 |
-
drapery_fullness: float = 1.5
|
287 |
|
288 |
def __post_init__(self):
|
289 |
super().__post_init__()
|
@@ -398,11 +383,6 @@ class ReferenceData:
|
|
398 |
"Reflective": {"u_value": 2.4, "shgc": 0.38, "glazing_type": "Reflective", "frame_type": "Aluminum with Thermal Break"},
|
399 |
"Custom": {"u_value": 3.0, "shgc": 0.7, "glazing_type": "Double Clear", "frame_type": "Aluminum with Thermal Break"}
|
400 |
},
|
401 |
-
"surface_colors": {
|
402 |
-
"Dark": 0.9,
|
403 |
-
"Medium": 0.6,
|
404 |
-
"Light": 0.3
|
405 |
-
},
|
406 |
"frame_types": {
|
407 |
"Aluminum without Thermal Break": 1.0,
|
408 |
"Aluminum with Thermal Break": 0.8,
|
@@ -516,11 +496,10 @@ class ComponentSelectionInterface:
|
|
516 |
name = st.text_input("Name", "New Wall")
|
517 |
area = st.number_input("Area (m²)", min_value=0.0, value=1.0, step=0.1)
|
518 |
orientation = st.selectbox("Orientation", [o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE], index=0)
|
519 |
-
surface_color = st.selectbox("Surface Color", ["Dark", "Medium", "Light"], index=0)
|
520 |
crack_length = st.number_input("Crack Length (m)", min_value=0.0, max_value=100.0, value=0.0, step=0.1)
|
521 |
with col2:
|
522 |
wall_options = self.reference_data.data["wall_types"]
|
523 |
-
selected_wall = st
|
524 |
u_value = st.number_input("U-Value (W/m²·K)", min_value=0.0, value=float(wall_options[selected_wall]["u_value"]), step=0.01)
|
525 |
wall_group = st.selectbox("Wall Group (ASHRAE)", ["A", "B", "C", "D", "E", "F", "G", "H"], index=0)
|
526 |
absorptivity = st.selectbox("Solar Absorptivity", ["Light (0.3)", "Light to Medium (0.45)", "Medium (0.6)", "Medium to Dark (0.75)", "Dark (0.9)"], index=2)
|
@@ -536,7 +515,7 @@ class ComponentSelectionInterface:
|
|
536 |
name=name, u_value=u_value, area=area, orientation=Orientation(orientation),
|
537 |
wall_type=selected_wall, wall_group=wall_group, absorptivity=absorptivity_value,
|
538 |
shading_coefficient=shading_coefficient, infiltration_rate_cfm=infiltration_rate,
|
539 |
-
|
540 |
)
|
541 |
self.component_library.add_component(new_wall)
|
542 |
session_state.components['walls'].append(new_wall)
|
@@ -553,12 +532,12 @@ class ComponentSelectionInterface:
|
|
553 |
uploaded_file = st.file_uploader("Upload Walls File", type=["csv", "xlsx"], key="wall_upload")
|
554 |
required_cols = [
|
555 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Orientation", "Wall Type", "Wall Group",
|
556 |
-
"Absorptivity", "Shading Coefficient", "Infiltration Rate (CFM)",
|
557 |
"Crack Length (m)", "Crack Width (m)"
|
558 |
]
|
559 |
template_data = pd.DataFrame(columns=required_cols)
|
560 |
template_data.loc[0] = [
|
561 |
-
"Example Wall", 10.0, 0.5, "North", "Brick Wall", "A", 0.6, 1.0, 0.0,
|
562 |
]
|
563 |
st.download_button(
|
564 |
label="Download Wall Template",
|
@@ -577,7 +556,7 @@ class ComponentSelectionInterface:
|
|
577 |
wall_group=str(row["Wall Group"]), absorptivity=float(row["Absorptivity"]),
|
578 |
shading_coefficient=float(row["Shading Coefficient"]),
|
579 |
infiltration_rate_cfm=float(row["Infiltration Rate (CFM)"]),
|
580 |
-
|
581 |
crack_width=float(row["Crack Width (m)"])
|
582 |
)
|
583 |
self.component_library.add_component(new_wall)
|
@@ -601,7 +580,6 @@ class ComponentSelectionInterface:
|
|
601 |
with col1:
|
602 |
name = st.text_input("Name", "New Roof")
|
603 |
area = st.number_input("Area (m²)", min_value=0.0, value=1.0, step=0.1)
|
604 |
-
surface_color = st.selectbox("Surface Color", ["Dark", "Medium", "Light"], index=0)
|
605 |
roof_height = st.number_input("Roof Height (m)", min_value=0.0, max_value=100.0, value=3.0, step=0.1)
|
606 |
with col2:
|
607 |
roof_options = self.reference_data.data["roof_types"]
|
@@ -618,8 +596,7 @@ class ComponentSelectionInterface:
|
|
618 |
new_roof = Roof(
|
619 |
name=name, u_value=u_value, area=area, orientation=Orientation.HORIZONTAL,
|
620 |
roof_type=selected_roof, roof_group=roof_group, slope=slope,
|
621 |
-
absorptivity=absorptivity_value,
|
622 |
-
roof_height=roof_height
|
623 |
)
|
624 |
self.component_library.add_component(new_roof)
|
625 |
session_state.components['roofs'].append(new_roof)
|
@@ -651,11 +628,11 @@ class ComponentSelectionInterface:
|
|
651 |
uploaded_file = st.file_uploader("Upload Roofs File", type=["csv", "xlsx"], key="roof_upload")
|
652 |
required_cols = [
|
653 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Roof Type", "Roof Group", "Slope",
|
654 |
-
"Absorptivity", "
|
655 |
]
|
656 |
template_data = pd.DataFrame(columns=required_cols)
|
657 |
template_data.loc[0] = [
|
658 |
-
"Example Roof", 10.0, 0.3, "Concrete Roof", "A", "Flat", 0.6,
|
659 |
]
|
660 |
st.download_button(
|
661 |
label="Download Roof Template",
|
@@ -672,8 +649,7 @@ class ComponentSelectionInterface:
|
|
672 |
name=str(row["Name"]), u_value=float(row["U-Value (W/m²·K)"]), area=float(row["Area (m²)"]),
|
673 |
orientation=Orientation.HORIZONTAL, roof_type=str(row["Roof Type"]),
|
674 |
roof_group=str(row["Roof Group"]), slope=str(row["Slope"]),
|
675 |
-
absorptivity=float(row["Absorptivity"]),
|
676 |
-
roof_height=float(row["Roof Height (m)"])
|
677 |
)
|
678 |
self.component_library.add_component(new_roof)
|
679 |
session_state.components['roofs'].append(new_roof)
|
@@ -1092,11 +1068,11 @@ class ComponentSelectionInterface:
|
|
1092 |
headers = {
|
1093 |
ComponentType.WALL: [
|
1094 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Orientation", "Wall Type", "Wall Group",
|
1095 |
-
"Absorptivity", "
|
1096 |
],
|
1097 |
ComponentType.ROOF: [
|
1098 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Roof Type", "Roof Group", "Slope",
|
1099 |
-
"Absorptivity", "
|
1100 |
],
|
1101 |
ComponentType.FLOOR: [
|
1102 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Floor Type", "Ground Contact",
|
@@ -1130,18 +1106,16 @@ class ComponentSelectionInterface:
|
|
1130 |
cols[4].write(comp.wall_type)
|
1131 |
cols[5].write(comp.wall_group)
|
1132 |
cols[6].write(comp.absorptivity)
|
1133 |
-
cols[7].write(comp.
|
1134 |
-
cols[8].write(comp.
|
1135 |
-
|
1136 |
-
edit_col = 10
|
1137 |
elif component_type == ComponentType.ROOF:
|
1138 |
cols[3].write(comp.roof_type)
|
1139 |
cols[4].write(comp.roof_group)
|
1140 |
cols[5].write(comp.slope)
|
1141 |
cols[6].write(comp.absorptivity)
|
1142 |
-
cols[7].write(comp.
|
1143 |
-
|
1144 |
-
edit_col = 9
|
1145 |
elif component_type == ComponentType.FLOOR:
|
1146 |
cols[3].write(comp.floor_type)
|
1147 |
cols[4].write("Yes" if comp.ground_contact else "No")
|
@@ -1209,7 +1183,6 @@ class ComponentSelectionInterface:
|
|
1209 |
name = st.text_input("Name", wall.name)
|
1210 |
area = st.number_input("Area (m²)", min_value=0.0, value=wall.area, step=0.1)
|
1211 |
orientation = st.selectbox("Orientation", [o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE], index=[o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE].index(wall.orientation.value))
|
1212 |
-
surface_color = st.selectbox("Surface Color", ["Dark", "Medium", "Light"], index=["Dark", "Medium", "Light"].index(wall.surface_color))
|
1213 |
crack_length = st.number_input("Crack Length (m)", min_value=0.0, max_value=100.0, value=wall.crack_length, step=0.1)
|
1214 |
with col2:
|
1215 |
wall_options = self.reference_data.data["wall_types"]
|
@@ -1221,30 +1194,32 @@ class ComponentSelectionInterface:
|
|
1221 |
infiltration_rate = st.number_input("Infiltration Rate (CFM)", min_value=0.0, value=wall.infiltration_rate_cfm, step=0.1)
|
1222 |
crack_width = st.number_input("Crack Width (m)", min_value=0.0, max_value=0.1, value=wall.crack_width, step=0.001)
|
1223 |
|
1224 |
-
|
1225 |
-
|
1226 |
-
|
1227 |
-
|
1228 |
-
|
1229 |
-
session_state[f"edit_wall"] = None
|
1230 |
-
st.rerun()
|
1231 |
|
1232 |
-
|
1233 |
-
|
1234 |
-
|
1235 |
-
|
1236 |
-
|
1237 |
-
|
1238 |
-
|
1239 |
-
|
1240 |
-
|
1241 |
-
|
1242 |
-
|
1243 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
1244 |
session_state[f"edit_wall"] = None
|
1245 |
st.rerun()
|
1246 |
-
except ValueError as e:
|
1247 |
-
st.error(f"Error: {str(e)}")
|
1248 |
|
1249 |
def _display_edit_roof_form(self, session_state: Any, roof: Roof, index: int) -> None:
|
1250 |
with st.form(f"edit_roof_form_{index}", clear_on_submit=True):
|
@@ -1252,7 +1227,6 @@ class ComponentSelectionInterface:
|
|
1252 |
with col1:
|
1253 |
name = st.text_input("Name", roof.name)
|
1254 |
area = st.number_input("Area (m²)", min_value=0.0, value=roof.area, step=0.1)
|
1255 |
-
surface_color = st.selectbox("Surface Color", ["Dark", "Medium", "Light"], index=["Dark", "Medium", "Light"].index(roof.surface_color))
|
1256 |
roof_height = st.number_input("Roof Height (m)", min_value=0.0, max_value=100.0, value=roof.roof_height, step=0.1)
|
1257 |
with col2:
|
1258 |
roof_options = self.reference_data.data["roof_types"]
|
@@ -1262,13 +1236,11 @@ class ComponentSelectionInterface:
|
|
1262 |
slope = st.selectbox("Roof Slope", ["Flat", "Low Slope", "Steep Slope"], index=["Flat", "Low Slope", "Steep Slope"].index(roof.slope))
|
1263 |
absorptivity = st.selectbox("Solar Absorptivity", ["Light (0.3)", "Light to Medium (0.45)", "Medium (0.6)", "Medium to Dark (0.75)", "Dark (0.9)"], index=["Light (0.3)", "Light to Medium (0.45)", "Medium (0.6)", "Medium to Dark (0.75)", "Dark (0.9)"].index(f"{['Light (0.3)', 'Light to Medium (0.45)', 'Medium (0.6)', 'Medium to Dark (0.75)', 'Dark (0.9)'][np.argmin([abs(float(opt.split('(')[1].strip(')')) - roof.absorptivity) for opt in ['Light (0.3)', 'Light to Medium (0.45)', 'Medium (0.6)', 'Medium to Dark (0.75)', 'Dark (0.9)']])]}"))
|
1264 |
|
1265 |
-
|
1266 |
-
with
|
1267 |
submitted = st.form_submit_button("Update Roof")
|
1268 |
-
with
|
1269 |
-
|
1270 |
-
session_state[f"edit_roof"] = None
|
1271 |
-
st.rerun()
|
1272 |
|
1273 |
if submitted:
|
1274 |
try:
|
@@ -1276,16 +1248,19 @@ class ComponentSelectionInterface:
|
|
1276 |
updated_roof = Roof(
|
1277 |
id=roof.id, name=name, u_value=u_value, area=area, orientation=Orientation.HORIZONTAL,
|
1278 |
roof_type=selected_roof, roof_group=roof_group, slope=slope,
|
1279 |
-
absorptivity=absorptivity_value,
|
1280 |
-
roof_height=roof_height
|
1281 |
)
|
1282 |
self.component_library.components[roof.id] = updated_roof
|
1283 |
session_state.components['roofs'][index] = updated_roof
|
1284 |
-
st.success(f"Updated {
|
1285 |
session_state[f"edit_roof"] = None
|
1286 |
st.rerun()
|
1287 |
except ValueError as e:
|
1288 |
st.error(f"Error: {str(e)}")
|
|
|
|
|
|
|
|
|
1289 |
|
1290 |
def _display_edit_floor_form(self, session_state: Any, floor: Floor, index: int) -> None:
|
1291 |
with st.form(f"edit_floor_form_{index}", clear_on_submit=True):
|
@@ -1302,13 +1277,11 @@ class ComponentSelectionInterface:
|
|
1302 |
ground_temp = st.number_input("Ground Temperature (°C)", min_value=-10.0, max_value=40.0, value=floor.ground_temperature_c, step=0.1) if ground_contact == "Yes" else floor.ground_temperature_c
|
1303 |
insulated = st.checkbox("Insulated Floor (e.g., R-10)", value=floor.insulated)
|
1304 |
|
1305 |
-
|
1306 |
-
with
|
1307 |
submitted = st.form_submit_button("Update Floor")
|
1308 |
-
with
|
1309 |
-
|
1310 |
-
session_state[f"edit_floor"] = None
|
1311 |
-
st.rerun()
|
1312 |
|
1313 |
if submitted:
|
1314 |
try:
|
@@ -1319,11 +1292,15 @@ class ComponentSelectionInterface:
|
|
1319 |
)
|
1320 |
self.component_library.components[floor.id] = updated_floor
|
1321 |
session_state.components['floors'][index] = updated_floor
|
1322 |
-
st.success(f"Updated {
|
1323 |
session_state[f"edit_floor"] = None
|
1324 |
st.rerun()
|
1325 |
except ValueError as e:
|
1326 |
st.error(f"Error: {str(e)}")
|
|
|
|
|
|
|
|
|
1327 |
|
1328 |
def _display_edit_window_form(self, session_state: Any, window: Window, index: int) -> None:
|
1329 |
with st.form(f"edit_window_form_{index}", clear_on_submit=True):
|
@@ -1333,7 +1310,7 @@ class ComponentSelectionInterface:
|
|
1333 |
area = st.number_input("Area (m²)", min_value=0.0, value=window.area, step=0.1)
|
1334 |
orientation = st.selectbox("Orientation", [o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE], index=[o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE].index(window.orientation.value))
|
1335 |
window_options = self.reference_data.data["window_types"]
|
1336 |
-
selected_window = st.selectbox("Window Type", options=list(window_options.keys()), index=list(window_options.keys()).index(window.
|
1337 |
glazing_type = st.selectbox("Glazing Type", [g.value for g in GlazingType], index=[g.value for g in GlazingType].index(window.glazing_type))
|
1338 |
drapery_openness = st.selectbox("Drapery Openness", [o.value for o in DraperyOpenness], index=[o.value for o in DraperyOpenness].index(window.drapery_openness))
|
1339 |
with col2:
|
@@ -1343,19 +1320,17 @@ class ComponentSelectionInterface:
|
|
1343 |
frame_percentage = st.slider("Frame Percentage (%)", min_value=0.0, max_value=30.0, value=window.frame_percentage)
|
1344 |
shading_options = {f"{k} (SC={v})": k for k, v in self.reference_data.data["shading_devices"].items()}
|
1345 |
shading_options["Custom"] = "Custom"
|
1346 |
-
shading_device = st.selectbox("Shading Device", options=list(shading_options.keys()), index=list(shading_options.values()).index(window.shading_device) if window.shading_device in shading_options.values() else
|
1347 |
shading_coefficient = st.number_input("Custom Shading Coefficient", min_value=0.0, max_value=1.0, value=window.shading_coefficient, step=0.05) if shading_device == "Custom" else self.reference_data.data["shading_devices"][shading_options[shading_device]]
|
1348 |
infiltration_rate = st.number_input("Infiltration Rate (CFM)", min_value=0.0, value=window.infiltration_rate_cfm, step=0.1)
|
1349 |
drapery_color = st.selectbox("Drapery Color", [c.value for c in DraperyColor], index=[c.value for c in DraperyColor].index(window.drapery_color))
|
1350 |
drapery_fullness = st.number_input("Drapery Fullness", min_value=1.0, max_value=2.0, value=window.drapery_fullness, step=0.1)
|
1351 |
|
1352 |
-
|
1353 |
-
with
|
1354 |
submitted = st.form_submit_button("Update Window")
|
1355 |
-
with
|
1356 |
-
|
1357 |
-
session_state[f"edit_window"] = None
|
1358 |
-
st.rerun()
|
1359 |
|
1360 |
if submitted:
|
1361 |
try:
|
@@ -1368,11 +1343,15 @@ class ComponentSelectionInterface:
|
|
1368 |
)
|
1369 |
self.component_library.components[window.id] = updated_window
|
1370 |
session_state.components['windows'][index] = updated_window
|
1371 |
-
st.success(f"Updated {
|
1372 |
session_state[f"edit_window"] = None
|
1373 |
st.rerun()
|
1374 |
except ValueError as e:
|
1375 |
st.error(f"Error: {str(e)}")
|
|
|
|
|
|
|
|
|
1376 |
|
1377 |
def _display_edit_door_form(self, session_state: Any, door: Door, index: int) -> None:
|
1378 |
with st.form(f"edit_door_form_{index}", clear_on_submit=True):
|
@@ -1389,13 +1368,11 @@ class ComponentSelectionInterface:
|
|
1389 |
infiltration_rate = st.number_input("Infiltration Rate (CFM)", min_value=0.0, value=door.infiltration_rate_cfm, step=0.1)
|
1390 |
crack_width = st.number_input("Crack Width (m)", min_value=0.0, max_value=0.1, value=door.crack_width, step=0.001)
|
1391 |
|
1392 |
-
|
1393 |
-
with
|
1394 |
submitted = st.form_submit_button("Update Door")
|
1395 |
-
with
|
1396 |
-
|
1397 |
-
session_state[f"edit_door"] = None
|
1398 |
-
st.rerun()
|
1399 |
|
1400 |
if submitted:
|
1401 |
try:
|
@@ -1406,11 +1383,15 @@ class ComponentSelectionInterface:
|
|
1406 |
)
|
1407 |
self.component_library.components[door.id] = updated_door
|
1408 |
session_state.components['doors'][index] = updated_door
|
1409 |
-
st.success(f"Updated {
|
1410 |
session_state[f"edit_door"] = None
|
1411 |
st.rerun()
|
1412 |
except ValueError as e:
|
1413 |
st.error(f"Error: {str(e)}")
|
|
|
|
|
|
|
|
|
1414 |
|
1415 |
def _display_edit_skylight_form(self, session_state: Any, skylight: Skylight, index: int) -> None:
|
1416 |
with st.form(f"edit_skylight_form_{index}", clear_on_submit=True):
|
@@ -1419,7 +1400,7 @@ class ComponentSelectionInterface:
|
|
1419 |
name = st.text_input("Name", skylight.name)
|
1420 |
area = st.number_input("Area (m²)", min_value=0.0, value=skylight.area, step=0.1)
|
1421 |
skylight_options = self.reference_data.data["skylight_types"]
|
1422 |
-
selected_skylight = st.selectbox("Skylight Type", options=list(skylight_options.keys()), index=list(skylight_options.keys()).index(skylight.
|
1423 |
glazing_type = st.selectbox("Glazing Type", [g.value for g in GlazingType], index=[g.value for g in GlazingType].index(skylight.glazing_type))
|
1424 |
drapery_openness = st.selectbox("Drapery Openness", [o.value for o in DraperyOpenness], index=[o.value for o in DraperyOpenness].index(skylight.drapery_openness))
|
1425 |
with col2:
|
@@ -1429,19 +1410,17 @@ class ComponentSelectionInterface:
|
|
1429 |
frame_percentage = st.slider("Frame Percentage (%)", min_value=0.0, max_value=30.0, value=skylight.frame_percentage)
|
1430 |
shading_options = {f"{k} (SC={v})": k for k, v in self.reference_data.data["shading_devices"].items()}
|
1431 |
shading_options["Custom"] = "Custom"
|
1432 |
-
shading_device = st.selectbox("Shading Device", options=list(shading_options.keys()), index=list(shading_options.values()).index(skylight.shading_device) if skylight.shading_device in shading_options.values() else
|
1433 |
shading_coefficient = st.number_input("Custom Shading Coefficient", min_value=0.0, max_value=1.0, value=skylight.shading_coefficient, step=0.05) if shading_device == "Custom" else self.reference_data.data["shading_devices"][shading_options[shading_device]]
|
1434 |
infiltration_rate = st.number_input("Infiltration Rate (CFM)", min_value=0.0, value=skylight.infiltration_rate_cfm, step=0.1)
|
1435 |
drapery_color = st.selectbox("Drapery Color", [c.value for c in DraperyColor], index=[c.value for c in DraperyColor].index(skylight.drapery_color))
|
1436 |
drapery_fullness = st.number_input("Drapery Fullness", min_value=1.0, max_value=2.0, value=skylight.drapery_fullness, step=0.1)
|
1437 |
|
1438 |
-
|
1439 |
-
with
|
1440 |
submitted = st.form_submit_button("Update Skylight")
|
1441 |
-
with
|
1442 |
-
|
1443 |
-
session_state[f"edit_skylight"] = None
|
1444 |
-
st.rerun()
|
1445 |
|
1446 |
if submitted:
|
1447 |
try:
|
@@ -1454,38 +1433,38 @@ class ComponentSelectionInterface:
|
|
1454 |
)
|
1455 |
self.component_library.components[skylight.id] = updated_skylight
|
1456 |
session_state.components['skylights'][index] = updated_skylight
|
1457 |
-
st.success(f"Updated {
|
1458 |
session_state[f"edit_skylight"] = None
|
1459 |
st.rerun()
|
1460 |
except ValueError as e:
|
1461 |
st.error(f"Error: {str(e)}")
|
|
|
|
|
|
|
|
|
1462 |
|
1463 |
def _save_components(self, session_state: Any) -> None:
|
1464 |
components_data = {
|
1465 |
-
"walls": [comp.to_dict() for comp in session_state.components
|
1466 |
-
"roofs": [comp.to_dict() for comp in session_state.components
|
1467 |
-
"floors": [comp.to_dict() for comp in session_state.components
|
1468 |
-
"windows": [comp.to_dict() for comp in session_state.components
|
1469 |
-
"doors": [comp.to_dict() for comp in session_state.components
|
1470 |
-
"skylights": [comp.to_dict() for comp in session_state.components
|
1471 |
"roof_air_volume_m3": session_state.roof_air_volume_m3,
|
1472 |
"roof_ventilation_ach": session_state.roof_ventilation_ach
|
1473 |
}
|
1474 |
-
|
1475 |
-
json.dump(components_data,
|
1476 |
st.download_button(
|
1477 |
label="Download Components JSON",
|
1478 |
-
data=
|
1479 |
file_name="components.json",
|
1480 |
mime="application/json"
|
1481 |
)
|
1482 |
-
st.success("Components saved!
|
1483 |
-
|
1484 |
-
# --- Main Application ---
|
1485 |
-
def main():
|
1486 |
-
st.set_page_config(page_title="HVAC Component Selection", layout="wide")
|
1487 |
-
component_selection = ComponentSelectionInterface()
|
1488 |
-
component_selection.display_component_selection(st.session_state)
|
1489 |
|
|
|
1490 |
if __name__ == "__main__":
|
1491 |
-
|
|
|
|
4 |
All dependencies are included within this file for standalone operation.
|
5 |
Updated 2025-04-28: Added surface color selection, expanded component types, and skylight support.
|
6 |
Updated 2025-05-02: Added crack dimensions for walls/doors, drapery properties for windows/skylights, and roof height per enhancement plan.
|
7 |
+
Updated 2025-05-03: Removed surface_color field, using solar absorptivity (Light 0.3, Light to Medium 0.45, Medium 0.6, Medium to Dark 0.75, Dark 0.9) exclusively.
|
8 |
|
9 |
Author: Dr Majed Abuseif
|
10 |
"""
|
|
|
40 |
DOOR = "Door"
|
41 |
SKYLIGHT = "Skylight"
|
42 |
|
|
|
|
|
|
|
|
|
|
|
43 |
class GlazingType(Enum):
|
44 |
SINGLE_CLEAR = "Single Clear"
|
45 |
SINGLE_TINTED = "Single Tinted"
|
|
|
100 |
absorptivity: float = 0.6
|
101 |
shading_coefficient: float = 1.0
|
102 |
infiltration_rate_cfm: float = 0.0
|
|
|
103 |
crack_length: float = 0.0 # Added for infiltration (m)
|
104 |
crack_width: float = 0.0 # Added for infiltration (m)
|
105 |
|
|
|
120 |
if self.wall_group not in VALID_WALL_GROUPS:
|
121 |
st.warning(f"Invalid wall_group '{self.wall_group}' for wall '{self.name}'. Defaulting to 'A'.")
|
122 |
self.wall_group = "A"
|
|
|
|
|
|
|
|
|
123 |
|
124 |
def to_dict(self) -> dict:
|
125 |
base_dict = super().to_dict()
|
126 |
base_dict.update({
|
127 |
"wall_type": self.wall_type, "wall_group": self.wall_group, "absorptivity": self.absorptivity,
|
128 |
"shading_coefficient": self.shading_coefficient, "infiltration_rate_cfm": self.infiltration_rate_cfm,
|
129 |
+
"crack_length": self.crack_length, "crack_width": self.crack_width
|
130 |
})
|
131 |
return base_dict
|
132 |
|
|
|
136 |
roof_group: str = "A" # ASHRAE group
|
137 |
slope: str = "Flat"
|
138 |
absorptivity: float = 0.6
|
|
|
139 |
roof_height: float = 3.0 # Added for stack effect (m)
|
140 |
|
141 |
def __post_init__(self):
|
|
|
151 |
if self.roof_group not in VALID_ROOF_GROUPS:
|
152 |
st.warning(f"Invalid roof_group '{self.roof_group}' for roof '{self.name}'. Defaulting to 'A'.")
|
153 |
self.roof_group = "A"
|
|
|
|
|
|
|
|
|
154 |
|
155 |
def to_dict(self) -> dict:
|
156 |
base_dict = super().to_dict()
|
157 |
base_dict.update({
|
158 |
"roof_type": self.roof_type, "roof_group": self.roof_group, "slope": self.slope,
|
159 |
+
"absorptivity": self.absorptivity, "roof_height": self.roof_height
|
|
|
160 |
})
|
161 |
return base_dict
|
162 |
|
|
|
191 |
shgc: float = 0.7
|
192 |
shading_device: str = "None"
|
193 |
shading_coefficient: float = 1.0
|
194 |
+
frame_type: str = "Aluminum without Thermal Break"
|
195 |
frame_percentage: float = 20.0
|
196 |
infiltration_rate_cfm: float = 0.0
|
197 |
+
glazing_type: str = "Double Clear"
|
198 |
+
drapery_openness: str = "Open"
|
199 |
+
drapery_color: str = "Light"
|
200 |
+
drapery_fullness: float = 1.5
|
201 |
|
202 |
def __post_init__(self):
|
203 |
super().__post_init__()
|
|
|
236 |
class Door(BuildingComponent):
|
237 |
door_type: str = "Solid Wood"
|
238 |
infiltration_rate_cfm: float = 0.0
|
239 |
+
crack_length: float = 0.0
|
240 |
+
crack_width: float = 0.0
|
241 |
|
242 |
def __post_init__(self):
|
243 |
super().__post_init__()
|
|
|
262 |
shgc: float = 0.7
|
263 |
shading_device: str = "None"
|
264 |
shading_coefficient: float = 1.0
|
265 |
+
frame_type: str = "Aluminum without Thermal Break"
|
266 |
frame_percentage: float = 20.0
|
267 |
infiltration_rate_cfm: float = 0.0
|
268 |
+
glazing_type: str = "Double Clear"
|
269 |
+
drapery_openness: str = "Open"
|
270 |
+
drapery_color: str = "Light"
|
271 |
+
drapery_fullness: float = 1.5
|
272 |
|
273 |
def __post_init__(self):
|
274 |
super().__post_init__()
|
|
|
383 |
"Reflective": {"u_value": 2.4, "shgc": 0.38, "glazing_type": "Reflective", "frame_type": "Aluminum with Thermal Break"},
|
384 |
"Custom": {"u_value": 3.0, "shgc": 0.7, "glazing_type": "Double Clear", "frame_type": "Aluminum with Thermal Break"}
|
385 |
},
|
|
|
|
|
|
|
|
|
|
|
386 |
"frame_types": {
|
387 |
"Aluminum without Thermal Break": 1.0,
|
388 |
"Aluminum with Thermal Break": 0.8,
|
|
|
496 |
name = st.text_input("Name", "New Wall")
|
497 |
area = st.number_input("Area (m²)", min_value=0.0, value=1.0, step=0.1)
|
498 |
orientation = st.selectbox("Orientation", [o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE], index=0)
|
|
|
499 |
crack_length = st.number_input("Crack Length (m)", min_value=0.0, max_value=100.0, value=0.0, step=0.1)
|
500 |
with col2:
|
501 |
wall_options = self.reference_data.data["wall_types"]
|
502 |
+
selected_wall = st seletbox("Wall Type", options=list(wall_options.keys()))
|
503 |
u_value = st.number_input("U-Value (W/m²·K)", min_value=0.0, value=float(wall_options[selected_wall]["u_value"]), step=0.01)
|
504 |
wall_group = st.selectbox("Wall Group (ASHRAE)", ["A", "B", "C", "D", "E", "F", "G", "H"], index=0)
|
505 |
absorptivity = st.selectbox("Solar Absorptivity", ["Light (0.3)", "Light to Medium (0.45)", "Medium (0.6)", "Medium to Dark (0.75)", "Dark (0.9)"], index=2)
|
|
|
515 |
name=name, u_value=u_value, area=area, orientation=Orientation(orientation),
|
516 |
wall_type=selected_wall, wall_group=wall_group, absorptivity=absorptivity_value,
|
517 |
shading_coefficient=shading_coefficient, infiltration_rate_cfm=infiltration_rate,
|
518 |
+
crack_length=crack_length, crack_width=crack_width
|
519 |
)
|
520 |
self.component_library.add_component(new_wall)
|
521 |
session_state.components['walls'].append(new_wall)
|
|
|
532 |
uploaded_file = st.file_uploader("Upload Walls File", type=["csv", "xlsx"], key="wall_upload")
|
533 |
required_cols = [
|
534 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Orientation", "Wall Type", "Wall Group",
|
535 |
+
"Absorptivity", "Shading Coefficient", "Infiltration Rate (CFM)",
|
536 |
"Crack Length (m)", "Crack Width (m)"
|
537 |
]
|
538 |
template_data = pd.DataFrame(columns=required_cols)
|
539 |
template_data.loc[0] = [
|
540 |
+
"Example Wall", 10.0, 0.5, "North", "Brick Wall", "A", 0.6, 1.0, 0.0, 0.0, 0.0
|
541 |
]
|
542 |
st.download_button(
|
543 |
label="Download Wall Template",
|
|
|
556 |
wall_group=str(row["Wall Group"]), absorptivity=float(row["Absorptivity"]),
|
557 |
shading_coefficient=float(row["Shading Coefficient"]),
|
558 |
infiltration_rate_cfm=float(row["Infiltration Rate (CFM)"]),
|
559 |
+
crack_length=float(row["Crack Length (m)"]),
|
560 |
crack_width=float(row["Crack Width (m)"])
|
561 |
)
|
562 |
self.component_library.add_component(new_wall)
|
|
|
580 |
with col1:
|
581 |
name = st.text_input("Name", "New Roof")
|
582 |
area = st.number_input("Area (m²)", min_value=0.0, value=1.0, step=0.1)
|
|
|
583 |
roof_height = st.number_input("Roof Height (m)", min_value=0.0, max_value=100.0, value=3.0, step=0.1)
|
584 |
with col2:
|
585 |
roof_options = self.reference_data.data["roof_types"]
|
|
|
596 |
new_roof = Roof(
|
597 |
name=name, u_value=u_value, area=area, orientation=Orientation.HORIZONTAL,
|
598 |
roof_type=selected_roof, roof_group=roof_group, slope=slope,
|
599 |
+
absorptivity=absorptivity_value, roof_height=roof_height
|
|
|
600 |
)
|
601 |
self.component_library.add_component(new_roof)
|
602 |
session_state.components['roofs'].append(new_roof)
|
|
|
628 |
uploaded_file = st.file_uploader("Upload Roofs File", type=["csv", "xlsx"], key="roof_upload")
|
629 |
required_cols = [
|
630 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Roof Type", "Roof Group", "Slope",
|
631 |
+
"Absorptivity", "Roof Height (m)"
|
632 |
]
|
633 |
template_data = pd.DataFrame(columns=required_cols)
|
634 |
template_data.loc[0] = [
|
635 |
+
"Example Roof", 10.0, 0.3, "Concrete Roof", "A", "Flat", 0.6, 3.0
|
636 |
]
|
637 |
st.download_button(
|
638 |
label="Download Roof Template",
|
|
|
649 |
name=str(row["Name"]), u_value=float(row["U-Value (W/m²·K)"]), area=float(row["Area (m²)"]),
|
650 |
orientation=Orientation.HORIZONTAL, roof_type=str(row["Roof Type"]),
|
651 |
roof_group=str(row["Roof Group"]), slope=str(row["Slope"]),
|
652 |
+
absorptivity=float(row["Absorptivity"]), roof_height=float(row["Roof Height (m)"])
|
|
|
653 |
)
|
654 |
self.component_library.add_component(new_roof)
|
655 |
session_state.components['roofs'].append(new_roof)
|
|
|
1068 |
headers = {
|
1069 |
ComponentType.WALL: [
|
1070 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Orientation", "Wall Type", "Wall Group",
|
1071 |
+
"Absorptivity", "Crack Length (m)", "Crack Width (m)", "Edit", "Delete"
|
1072 |
],
|
1073 |
ComponentType.ROOF: [
|
1074 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Roof Type", "Roof Group", "Slope",
|
1075 |
+
"Absorptivity", "Roof Height (m)", "Edit", "Delete"
|
1076 |
],
|
1077 |
ComponentType.FLOOR: [
|
1078 |
"Name", "Area (m²)", "U-Value (W/m²·K)", "Floor Type", "Ground Contact",
|
|
|
1106 |
cols[4].write(comp.wall_type)
|
1107 |
cols[5].write(comp.wall_group)
|
1108 |
cols[6].write(comp.absorptivity)
|
1109 |
+
cols[7].write(comp.crack_length)
|
1110 |
+
cols[8].write(comp.crack_width)
|
1111 |
+
edit_col = 9
|
|
|
1112 |
elif component_type == ComponentType.ROOF:
|
1113 |
cols[3].write(comp.roof_type)
|
1114 |
cols[4].write(comp.roof_group)
|
1115 |
cols[5].write(comp.slope)
|
1116 |
cols[6].write(comp.absorptivity)
|
1117 |
+
cols[7].write(comp.roof_height)
|
1118 |
+
edit_col = 8
|
|
|
1119 |
elif component_type == ComponentType.FLOOR:
|
1120 |
cols[3].write(comp.floor_type)
|
1121 |
cols[4].write("Yes" if comp.ground_contact else "No")
|
|
|
1183 |
name = st.text_input("Name", wall.name)
|
1184 |
area = st.number_input("Area (m²)", min_value=0.0, value=wall.area, step=0.1)
|
1185 |
orientation = st.selectbox("Orientation", [o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE], index=[o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE].index(wall.orientation.value))
|
|
|
1186 |
crack_length = st.number_input("Crack Length (m)", min_value=0.0, max_value=100.0, value=wall.crack_length, step=0.1)
|
1187 |
with col2:
|
1188 |
wall_options = self.reference_data.data["wall_types"]
|
|
|
1194 |
infiltration_rate = st.number_input("Infiltration Rate (CFM)", min_value=0.0, value=wall.infiltration_rate_cfm, step=0.1)
|
1195 |
crack_width = st.number_input("Crack Width (m)", min_value=0.0, max_value=0.1, value=wall.crack_width, step=0.001)
|
1196 |
|
1197 |
+
col3, col4 = st.columns(2)
|
1198 |
+
with col3:
|
1199 |
+
submitted = st.form_submit_button("Update Wall")
|
1200 |
+
with col4:
|
1201 |
+
cancelled = st.form_submit_button("Cancel")
|
|
|
|
|
1202 |
|
1203 |
+
if submitted:
|
1204 |
+
try:
|
1205 |
+
absorptivity_value = float(absorptivity.split("(")[1].strip(")"))
|
1206 |
+
updated_wall = Wall(
|
1207 |
+
id=wall.id, name=name, u_value=u_value, area=area, orientation=Orientation(orientation),
|
1208 |
+
wall_type=selected_wall, wall_group=wall_group, absorptivity=absorptivity_value,
|
1209 |
+
shading_coefficient=shading_coefficient, infiltration_rate_cfm=infiltration_rate,
|
1210 |
+
crack_length=crack_length, crack_width=crack_width
|
1211 |
+
)
|
1212 |
+
self.component_library.components[wall.id] = updated_wall
|
1213 |
+
session_state.components['walls'][index] = updated_wall
|
1214 |
+
st.success(f"Updated {name}")
|
1215 |
+
session_state[f"edit_wall"] = None
|
1216 |
+
st.rerun()
|
1217 |
+
except ValueError as e:
|
1218 |
+
st.error(f"Error: {str(e)}")
|
1219 |
+
|
1220 |
+
if cancelled:
|
1221 |
session_state[f"edit_wall"] = None
|
1222 |
st.rerun()
|
|
|
|
|
1223 |
|
1224 |
def _display_edit_roof_form(self, session_state: Any, roof: Roof, index: int) -> None:
|
1225 |
with st.form(f"edit_roof_form_{index}", clear_on_submit=True):
|
|
|
1227 |
with col1:
|
1228 |
name = st.text_input("Name", roof.name)
|
1229 |
area = st.number_input("Area (m²)", min_value=0.0, value=roof.area, step=0.1)
|
|
|
1230 |
roof_height = st.number_input("Roof Height (m)", min_value=0.0, max_value=100.0, value=roof.roof_height, step=0.1)
|
1231 |
with col2:
|
1232 |
roof_options = self.reference_data.data["roof_types"]
|
|
|
1236 |
slope = st.selectbox("Roof Slope", ["Flat", "Low Slope", "Steep Slope"], index=["Flat", "Low Slope", "Steep Slope"].index(roof.slope))
|
1237 |
absorptivity = st.selectbox("Solar Absorptivity", ["Light (0.3)", "Light to Medium (0.45)", "Medium (0.6)", "Medium to Dark (0.75)", "Dark (0.9)"], index=["Light (0.3)", "Light to Medium (0.45)", "Medium (0.6)", "Medium to Dark (0.75)", "Dark (0.9)"].index(f"{['Light (0.3)', 'Light to Medium (0.45)', 'Medium (0.6)', 'Medium to Dark (0.75)', 'Dark (0.9)'][np.argmin([abs(float(opt.split('(')[1].strip(')')) - roof.absorptivity) for opt in ['Light (0.3)', 'Light to Medium (0.45)', 'Medium (0.6)', 'Medium to Dark (0.75)', 'Dark (0.9)']])]}"))
|
1238 |
|
1239 |
+
col3, col4 = st.columns(2)
|
1240 |
+
with col3:
|
1241 |
submitted = st.form_submit_button("Update Roof")
|
1242 |
+
with col4:
|
1243 |
+
cancelled = st.form_submit_button("Cancel")
|
|
|
|
|
1244 |
|
1245 |
if submitted:
|
1246 |
try:
|
|
|
1248 |
updated_roof = Roof(
|
1249 |
id=roof.id, name=name, u_value=u_value, area=area, orientation=Orientation.HORIZONTAL,
|
1250 |
roof_type=selected_roof, roof_group=roof_group, slope=slope,
|
1251 |
+
absorptivity=absorptivity_value, roof_height=roof_height
|
|
|
1252 |
)
|
1253 |
self.component_library.components[roof.id] = updated_roof
|
1254 |
session_state.components['roofs'][index] = updated_roof
|
1255 |
+
st.success(f"Updated {name}")
|
1256 |
session_state[f"edit_roof"] = None
|
1257 |
st.rerun()
|
1258 |
except ValueError as e:
|
1259 |
st.error(f"Error: {str(e)}")
|
1260 |
+
|
1261 |
+
if cancelled:
|
1262 |
+
session_state[f"edit_roof"] = None
|
1263 |
+
st.rerun()
|
1264 |
|
1265 |
def _display_edit_floor_form(self, session_state: Any, floor: Floor, index: int) -> None:
|
1266 |
with st.form(f"edit_floor_form_{index}", clear_on_submit=True):
|
|
|
1277 |
ground_temp = st.number_input("Ground Temperature (°C)", min_value=-10.0, max_value=40.0, value=floor.ground_temperature_c, step=0.1) if ground_contact == "Yes" else floor.ground_temperature_c
|
1278 |
insulated = st.checkbox("Insulated Floor (e.g., R-10)", value=floor.insulated)
|
1279 |
|
1280 |
+
col3, col4 = st.columns(2)
|
1281 |
+
with col3:
|
1282 |
submitted = st.form_submit_button("Update Floor")
|
1283 |
+
with col4:
|
1284 |
+
cancelled = st.form_submit_button("Cancel")
|
|
|
|
|
1285 |
|
1286 |
if submitted:
|
1287 |
try:
|
|
|
1292 |
)
|
1293 |
self.component_library.components[floor.id] = updated_floor
|
1294 |
session_state.components['floors'][index] = updated_floor
|
1295 |
+
st.success(f"Updated {name}")
|
1296 |
session_state[f"edit_floor"] = None
|
1297 |
st.rerun()
|
1298 |
except ValueError as e:
|
1299 |
st.error(f"Error: {str(e)}")
|
1300 |
+
|
1301 |
+
if cancelled:
|
1302 |
+
session_state[f"edit_floor"] = None
|
1303 |
+
st.rerun()
|
1304 |
|
1305 |
def _display_edit_window_form(self, session_state: Any, window: Window, index: int) -> None:
|
1306 |
with st.form(f"edit_window_form_{index}", clear_on_submit=True):
|
|
|
1310 |
area = st.number_input("Area (m²)", min_value=0.0, value=window.area, step=0.1)
|
1311 |
orientation = st.selectbox("Orientation", [o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE], index=[o.value for o in Orientation if o != Orientation.HORIZONTAL and o != Orientation.NOT_APPLICABLE].index(window.orientation.value))
|
1312 |
window_options = self.reference_data.data["window_types"]
|
1313 |
+
selected_window = st.selectbox("Window Type", options=list(window_options.keys()), index=list(window_options.keys()).index(window.glazing_type) if window.glazing_type in window_options else 0)
|
1314 |
glazing_type = st.selectbox("Glazing Type", [g.value for g in GlazingType], index=[g.value for g in GlazingType].index(window.glazing_type))
|
1315 |
drapery_openness = st.selectbox("Drapery Openness", [o.value for o in DraperyOpenness], index=[o.value for o in DraperyOpenness].index(window.drapery_openness))
|
1316 |
with col2:
|
|
|
1320 |
frame_percentage = st.slider("Frame Percentage (%)", min_value=0.0, max_value=30.0, value=window.frame_percentage)
|
1321 |
shading_options = {f"{k} (SC={v})": k for k, v in self.reference_data.data["shading_devices"].items()}
|
1322 |
shading_options["Custom"] = "Custom"
|
1323 |
+
shading_device = st.selectbox("Shading Device", options=list(shading_options.keys()), index=list(shading_options.values()).index(window.shading_device) if window.shading_device in shading_options.values() else 0)
|
1324 |
shading_coefficient = st.number_input("Custom Shading Coefficient", min_value=0.0, max_value=1.0, value=window.shading_coefficient, step=0.05) if shading_device == "Custom" else self.reference_data.data["shading_devices"][shading_options[shading_device]]
|
1325 |
infiltration_rate = st.number_input("Infiltration Rate (CFM)", min_value=0.0, value=window.infiltration_rate_cfm, step=0.1)
|
1326 |
drapery_color = st.selectbox("Drapery Color", [c.value for c in DraperyColor], index=[c.value for c in DraperyColor].index(window.drapery_color))
|
1327 |
drapery_fullness = st.number_input("Drapery Fullness", min_value=1.0, max_value=2.0, value=window.drapery_fullness, step=0.1)
|
1328 |
|
1329 |
+
col3, col4 = st.columns(2)
|
1330 |
+
with col3:
|
1331 |
submitted = st.form_submit_button("Update Window")
|
1332 |
+
with col4:
|
1333 |
+
cancelled = st.form_submit_button("Cancel")
|
|
|
|
|
1334 |
|
1335 |
if submitted:
|
1336 |
try:
|
|
|
1343 |
)
|
1344 |
self.component_library.components[window.id] = updated_window
|
1345 |
session_state.components['windows'][index] = updated_window
|
1346 |
+
st.success(f"Updated {name}")
|
1347 |
session_state[f"edit_window"] = None
|
1348 |
st.rerun()
|
1349 |
except ValueError as e:
|
1350 |
st.error(f"Error: {str(e)}")
|
1351 |
+
|
1352 |
+
if cancelled:
|
1353 |
+
session_state[f"edit_window"] = None
|
1354 |
+
st.rerun()
|
1355 |
|
1356 |
def _display_edit_door_form(self, session_state: Any, door: Door, index: int) -> None:
|
1357 |
with st.form(f"edit_door_form_{index}", clear_on_submit=True):
|
|
|
1368 |
infiltration_rate = st.number_input("Infiltration Rate (CFM)", min_value=0.0, value=door.infiltration_rate_cfm, step=0.1)
|
1369 |
crack_width = st.number_input("Crack Width (m)", min_value=0.0, max_value=0.1, value=door.crack_width, step=0.001)
|
1370 |
|
1371 |
+
col3, col4 = st.columns(2)
|
1372 |
+
with col3:
|
1373 |
submitted = st.form_submit_button("Update Door")
|
1374 |
+
with col4:
|
1375 |
+
cancelled = st.form_submit_button("Cancel")
|
|
|
|
|
1376 |
|
1377 |
if submitted:
|
1378 |
try:
|
|
|
1383 |
)
|
1384 |
self.component_library.components[door.id] = updated_door
|
1385 |
session_state.components['doors'][index] = updated_door
|
1386 |
+
st.success(f"Updated {name}")
|
1387 |
session_state[f"edit_door"] = None
|
1388 |
st.rerun()
|
1389 |
except ValueError as e:
|
1390 |
st.error(f"Error: {str(e)}")
|
1391 |
+
|
1392 |
+
if cancelled:
|
1393 |
+
session_state[f"edit_door"] = None
|
1394 |
+
st.rerun()
|
1395 |
|
1396 |
def _display_edit_skylight_form(self, session_state: Any, skylight: Skylight, index: int) -> None:
|
1397 |
with st.form(f"edit_skylight_form_{index}", clear_on_submit=True):
|
|
|
1400 |
name = st.text_input("Name", skylight.name)
|
1401 |
area = st.number_input("Area (m²)", min_value=0.0, value=skylight.area, step=0.1)
|
1402 |
skylight_options = self.reference_data.data["skylight_types"]
|
1403 |
+
selected_skylight = st.selectbox("Skylight Type", options=list(skylight_options.keys()), index=list(skylight_options.keys()).index(skylight.glazing_type) if skylight.glazing_type in skylight_options else 0)
|
1404 |
glazing_type = st.selectbox("Glazing Type", [g.value for g in GlazingType], index=[g.value for g in GlazingType].index(skylight.glazing_type))
|
1405 |
drapery_openness = st.selectbox("Drapery Openness", [o.value for o in DraperyOpenness], index=[o.value for o in DraperyOpenness].index(skylight.drapery_openness))
|
1406 |
with col2:
|
|
|
1410 |
frame_percentage = st.slider("Frame Percentage (%)", min_value=0.0, max_value=30.0, value=skylight.frame_percentage)
|
1411 |
shading_options = {f"{k} (SC={v})": k for k, v in self.reference_data.data["shading_devices"].items()}
|
1412 |
shading_options["Custom"] = "Custom"
|
1413 |
+
shading_device = st.selectbox("Shading Device", options=list(shading_options.keys()), index=list(shading_options.values()).index(skylight.shading_device) if skylight.shading_device in shading_options.values() else 0)
|
1414 |
shading_coefficient = st.number_input("Custom Shading Coefficient", min_value=0.0, max_value=1.0, value=skylight.shading_coefficient, step=0.05) if shading_device == "Custom" else self.reference_data.data["shading_devices"][shading_options[shading_device]]
|
1415 |
infiltration_rate = st.number_input("Infiltration Rate (CFM)", min_value=0.0, value=skylight.infiltration_rate_cfm, step=0.1)
|
1416 |
drapery_color = st.selectbox("Drapery Color", [c.value for c in DraperyColor], index=[c.value for c in DraperyColor].index(skylight.drapery_color))
|
1417 |
drapery_fullness = st.number_input("Drapery Fullness", min_value=1.0, max_value=2.0, value=skylight.drapery_fullness, step=0.1)
|
1418 |
|
1419 |
+
col3, col4 = st.columns(2)
|
1420 |
+
with col3:
|
1421 |
submitted = st.form_submit_button("Update Skylight")
|
1422 |
+
with col4:
|
1423 |
+
cancelled = st.form_submit_button("Cancel")
|
|
|
|
|
1424 |
|
1425 |
if submitted:
|
1426 |
try:
|
|
|
1433 |
)
|
1434 |
self.component_library.components[skylight.id] = updated_skylight
|
1435 |
session_state.components['skylights'][index] = updated_skylight
|
1436 |
+
st.success(f"Updated {name}")
|
1437 |
session_state[f"edit_skylight"] = None
|
1438 |
st.rerun()
|
1439 |
except ValueError as e:
|
1440 |
st.error(f"Error: {str(e)}")
|
1441 |
+
|
1442 |
+
if cancelled:
|
1443 |
+
session_state[f"edit_skylight"] = None
|
1444 |
+
st.rerun()
|
1445 |
|
1446 |
def _save_components(self, session_state: Any) -> None:
|
1447 |
components_data = {
|
1448 |
+
"walls": [comp.to_dict() for comp in session_state.components.get('walls', [])],
|
1449 |
+
"roofs": [comp.to_dict() for comp in session_state.components.get('roofs', [])],
|
1450 |
+
"floors": [comp.to_dict() for comp in session_state.components.get('floors', [])],
|
1451 |
+
"windows": [comp.to_dict() for comp in session_state.components.get('windows', [])],
|
1452 |
+
"doors": [comp.to_dict() for comp in session_state.components.get('doors', [])],
|
1453 |
+
"skylights": [comp.to_dict() for comp in session_state.components.get('skylights', [])],
|
1454 |
"roof_air_volume_m3": session_state.roof_air_volume_m3,
|
1455 |
"roof_ventilation_ach": session_state.roof_ventilation_ach
|
1456 |
}
|
1457 |
+
output = io.StringIO()
|
1458 |
+
json.dump(components_data, output, indent=4)
|
1459 |
st.download_button(
|
1460 |
label="Download Components JSON",
|
1461 |
+
data=output.getvalue(),
|
1462 |
file_name="components.json",
|
1463 |
mime="application/json"
|
1464 |
)
|
1465 |
+
st.success("Components saved! Click the button to download the JSON file.")
|
|
|
|
|
|
|
|
|
|
|
|
|
1466 |
|
1467 |
+
# --- Main Execution ---
|
1468 |
if __name__ == "__main__":
|
1469 |
+
interface = ComponentSelectionInterface()
|
1470 |
+
interface.display_component_selection(st.session_state)
|