Python Scripting in Tas

Python Scripting in Tas

In previous blog posts, we’ve looked at how you can get started with the Tas API. We mentioned that you can use almost any programing language to automate Tas, and I personally recommended getting started with Visual Basic for Applications (VBA) with Microsoft Excel. This is for two reasons; the first, is that an Excel worksheet is a convenient way to store data you might want to put into Tas and get out of Tas. The second, is that using Excel to write Macros requires very little to get started and, most importantly of all, comes with ‘Intellisense’

Intellisense menu for the Tas type library
When you add a period (.) to Tas variables in the visual basic editor, the Intellisense window appears telling you what operations are available for that object

This Intellisense, sometimes known as code completion or auto-complete, is pretty handy as it saves you the trouble of having to refer to online documention and type library information viewers as frequently when you are programming using our API.

With most Python setups, this Intellisense for Tas is not available, so in this blog post, we’ll walk through how to set it up.

Screenshot of VSCode showing intellisense for python using the tas API.
A screenshot of a python IDE showing intellisense for the Tas type libaries.

Which tools should I use?

For this example, we will be using Python 3 in conjunction with Visual Studio Code as our IDE. You can also use Pycharm or Visual Studio, but I have chosen Visual Studio Code as it’s free and is relatively lightweight and requires minimal setup.

Step 1: Install Python

If you haven’t already installed Python, you can do so from python.org. I have installed Python version 3.13.1, but any version of Python 3 should work.

Step 2: Install Visual Studio Code

You can install Visual Studio Code from here. Once you have installed it, you’ll want to install the Python plugin. The first time you open a .py file in visual studio code, it will ask you if you want to install the Python plugin. Alternatively, you can just search from it in the online Extensions library/

Step 3: Install comtypes using pip

After you have installed Python, you’ll need to install the comtypes Python library. This library can be used to automatically create Python bindings for the Tas type libraries, which is how the Visual Studio Python interpreter learns about the Tas API. 

Open a python terminal by pressing the Start button and searching for Python:

With the Python shell open, enter the following code, pressing return after each statement, in order to determine where the Python executable is installed on your machine:

				
					import os, sys
os.path.dirname(sys.executable)
				
			

You should end up with something like the following:

We can see from the above that on my computer, Python is installed to C:\users\User\AppData\Local\Programs\Python\Python313. We will need to use this directory in order to install comtypes using pip.

Open a command prompt and ‘change directory’ to the directory identified in the previous step:

Change directory using the cd command:

Change into the Scripts directory as per the above, and then you can run the pip install command in order to install comtypes:

				
					pip install comtypes
				
			

Step 4. Generate Bindings

Now we can use comtypes to generate bindings for the Tas type libraries. We will do this using a Python script which we should only have to run once, though it may need re-running from time to time if the bindings are stored in a temporary directory on your computer. Create a new file called GenerateBindings.py in visual studio code, and enter the following script:

				
					import os
import comtypes.client

# Define the directory containing the type libraries
type_lib_dir = r'C:\Program Files\Environmental Design Solutions Ltd\Tas'

# List of type library filenames
type_lib_files = ['TAS3D.tlb', 'tbd.tlb', 'tpd.tlb', 'twd.tlb', 'tai.tlb', 'tcr.tlb', 'tcd.tlb', 'tsd.tlb']

# Generate COM bindings for each type library
for tlb_file in type_lib_files:
    tlb_path = os.path.join(type_lib_dir, tlb_file)
    comtypes.client.GetModule(tlb_path)

print("COM bindings generated successfully for all type libraries.")

				
			

When you run this script, the bindings will be created immediately.

Step 5: Write a script to check it worked

Now we can write a script to check it worked. As a simple example, create a new file in visual studio code called HelloTas.py and enter the following:

				
					import comtypes.client
from typing import cast

# Initialise COM (optional but recommended)
comtypes.CoInitialize()

# Import the generated TAS3D library
from comtypes.gen import TAS3D  

# Create the COM object with the specific interface
tas3d_dynamic = comtypes.client.CreateObject("TAS3D.T3DDocument", interface=TAS3D.ITAS3D)

# Cast to the correct interface for IntelliSense
tas3d: TAS3D.ITAS3D = cast(TAS3D.ITAS3D, tas3d_dynamic)

# Now, IntelliSense should recognise properties and methods
buildingName = tas3d.Building.buildingName

#Print the building name to the console
print(buildingName)
				
			

You should notice that when you start entering the above, the Intellisense menu appears and gives you code suggestions:

Congratulations! You’re ready to write some Tas Python scripts. All of the scripts mentioned in the coding lesson blog posts so far can be converted to Python, so it might be a good idea to use these for inspiration when it comes to getting started!

CO₂ Generation Rates

Calculating Metabolic Pollutant Generation Rates (CO)

There are many regulations which endorse modelling occupant pollutant generation accumulation within buildings. For occupants, this pertains to carbon dioxide (CO2). Most standards set thresholds which should not be exceeded:

  • ASHRAE Standard 62.1
  • CIBSE Guide A
  • WELL Building Standard
  • LEED 
  • BREEAM 

Building simulation software often takes the CO2 generation rate as an input to the calculation, but modellers often are presented with a simple ‘number of occupants’ to base their assumptions on when it comes to occupancy gains and their pollutant generation rates. 

In this blog post, we’ll explore the relationship between ‘number of occupants’ and ‘CO2 generation rate’, and produce a basic calculator which can be used to aid calculating CO2 generation rates depending on the information you have available.

Why do our bodies produce CO2?

In order to convert food into energy, our bodies convert glucose and oxygen into energy, carbon dioxide and water. This process is known as respiration.

As most of us don’t just eat sugar (glucose), our bodies also have to convert things like proteins and carbohydrates into glucose at the start of the process. 

Converting different foods into glucose also requires energy, which is why metabolising different foods produces varying amounts of CO2. 

The equation for respiration is:

\( \text{C}_6\text{H}_{12}\text{O}_6 + 6 \, \text{O}_2 \rightarrow 6 \, \text{CO}_2 + 6 \, \text{H}_2\text{O} + \text{Energy (ATP)} \)

That is, glucose + oxygen -> carbon dioxide + water

sugar plus oxygen are used in respiration to produce water, carbon dioxide and metabolic energy.

What affects CO2 Generation Rates?

If you selected two random people from different age groups, genders and level of fitness and measured how much carbon dioxide these people expelled over the course of an hour, you would probably find that the amount of CO2 produced can vary quite significantly. This is because the amount of CO2 a person produces depends heavily on their diet and metabolism:

  • Metabolic Rate: This is the rate at which your body burns calories at rest. Higher metabolic rate = more CO2.
  • Physical activity level: When we exercise, we burn more calories and produce more CO2.
  • Diet: Different food gets metabolised into different amounts of CO2 due to different metabolic pathways. Eating more protein can produce more CO2 compared to carbohydrates. 
  • Body size & composition: An increase in body weight is usually associated with a higher metabolic rate
  • Age and gender: Metabolic rate usually declines with age. Men tend to have higher metabolic rates than women.
  • Environmental factors: Temperature and altitude can affect how our body produces energy (and CO2 as a biproduct).
  • Health conditions: e.g. hyperthyroidism can increase metabolic rate and therefore CO2 production.

Respiratory Quotient (RQ)

From the respiration equation mentioned earlier, we can see that the amount of CO2 a person produces is directly related to how much oxygen they consume during respiration.

The ratio of CO2 relesed to O2 consumed is known as the Respiratory Quotient (RQ):

\( \text{RQ} = \frac{\text{Volume of } \text{CO}_2 \, \text{produced}}{\text{Volume of } \text{O}_2 \, \text{consumed}} \)

We mentioned earlier that different foods are converted to glucose in order for respiration to take place. As this conversion, known as gluconeogenesis, relies on energy to take place, this implies that different foods consumed result in different amounts of CO2 being produced during metabolism. Therefore, the RQ varies depending on diet, and some example figures are given in the tables below.

Converting number of people to CO2 generation rates

In order to derive a relation between the number of people in a space and the amount of CO2 produced by each person per hour, we will use the following relation:

“For every 1 litre of oxygen the body uses, approximately 5 calories are expended”

Calories are a measure of how much energy we can consume during respiration, so therefore if we calculate how much oxygen a person consumes, we can use the Respiratory Quotient to determine how much CO2 is produced as a biproduct. 

We can relate kcals to Joules with the following relation:

  • \( 1 \, \text{kcal} = 4184 \, \text{Joules (J)} \)

And Watts, which we often use in building simulation software as an input, are related to joules by:

  • \( 1 \, \text{Watt (W)} = 1 \, \frac{\text{Joule (J)}}{\text{Second (s)}} \)

Putting these relations together, we can deduce:

  • \( 1 \, \text{litre O}_2/\text{min} \times 5 \, \text{kcal}/\text{litre O}_2 \times 4184 \, \text{J}/\text{kcal} = 20920 \, \text{J}/\text{min} \)

Converting to Joules per second (Watts):

  • \( \frac{20920 \, \text{J}/\text{min}}{60 \, \text{sec}/\text{min}} = 348.67 \, \text{J}/\text{sec} = 348.67 \, \text{Watts} \)

For 1 ml O2/ minute:

\( \frac{348.67 \, \text{Watts}}{1000 \, \text{ml}/\text{litre}} = 0.34867 \, \text{Watts}/\text{ml}/\text{O}_2/\text{min} \)

So therefore:

\( \text{Metabolic Rate (W)} = \text{VO}_2 \, (\text{ml}/\text{min}) \times 0.34867 \)

Rearranging so that we can calculate VO2:

\( \text{VO}_2 \, (\text{ml}/\text{min}) = \frac{\text{Metabolic Rate (W)}}{0.34867} \)

And converting to CO2 by multiplying by the Respiratory Quotient (RQ):

\( \text{CO}_2 \, (\text{ml}/\text{min}) = \frac{\text{Metabolic Rate (W)}}{0.34867} \times \text{RQ} \)

Converting to litres/hour:

\( \text{CO}_2 \, (\text{L}/\text{hour}) = \left( \frac{\text{Metabolic Rate (W)}}{0.34867} \right) \times \text{RQ} \times \frac{60}{1000} \)

Example Metabolic Rates

This table lists some rate of heat emissions for mixtures of males/females. 

Degree of Activity Typical Building Total Rate of Heat Emission (W) Sensible Heat (W) Latent Heat (W)
Seated at theatre Theatre, cinema (matinee) 95 65 30
Seated at theatre, night Theatre, cinema (night) 105 70 35
Seated, very light work Offices, hotels, apartments 115 70 45
Moderate office work Offices, hotels, apartments 130 75 55
Standing, light work; walking Department store, retail store 130 75 55
Walking; standing Bank 145 75 70
Sedentary work Restaurant 160 80 80
Light bench work Factory 220 80 140
Moderate dancing Dance hall 250 90 160
Walking; light machine work Factory 295 110 185
Bowling Bowling alley 425 170 255
Heavy work Factory 425 170 255
Heavy machine work; lifting Factory 470 185 285
Athletics Gymnasium 525 210 315

Source: ASHRAE Handbook: Fundamentals (2001)

Example Respiratory Quotients (RQ)

Substrate Respiratory Quotient (RQ)
Carbohydrate 1.00
Fat 0.696
Protein 0.818
Non Dairy 0.86
Mixed Diet 0.8

Source: Exercise Physiology. Nutrition, Energy, and Human Performance by William D. McArdle, Frank I. Katch, Victor L. Katch  & Wikipedia 

Putting it all together: Example

Let’s calculate the CO2 pollutant generation rate for an office with 1 occupant and a floor area of 3m2. Assumptions:

  • Metabolic rate 130W (Moderate office work)
  • Mixed diet (RQ = 0.8)
The calculation is therefore:
 
\( \text{CO}_2 \, (\text{L}/\text{hour}) = \left( \frac{130}{0.34867} \right) \times 0.8 \times \frac{60}{1000} = 17.897 \)
 
And to convert to litres / hour / m2, divide by the floor area:
 
\( \text{CO}_2 \, (\text{L}/\text{hour}/\text{m}^2) = \frac{\text{CO}_2 \, (\text{L}/\text{hour})}{\text{Area} \, (\text{m}^2)} = \frac{178.97}{3} = 5.97 \, \text{L}/\text{hour}/\text{m}^2 \)

Carbon Dioxide Calculator

To save time, you can use the following handy carbon dioxide pollutant generate calculator to help calculate pollutant generation rates for use in Tas:

This calculator uses the methods discussed above to equate number of occupants to CO2 pollutant generation rates. If you would like us to add more activity levels to this calculator, please get in touch. As metabolic rates vary so significantly between individuals, we would always recommend performing sensitivity analysis to ensure your buildings perform well given the inherent uncertainty in the pollutant generation rates. 

Other useful conversions

In building simulation software, we often work with Watts as our primary unit for metabolic rates as these directly relate to the amount of sensible and latent heat gain occupants emit into a space. In other industries, however, it is quite common to work with METs.

What is a MET?

1 MET is the rate of energy expenditure at rest, approximately equal to 1 kcal per kg of body weight per hour, or 3.5 mL of oxygen consumed per kg of body weight per minute

Converting METs to Watts

Converting METs to Watts is fairly straightforward if you know the body weight of the individual:

\( \text{Watts (W)} = \text{MET} \times \text{Body Weight (kg)} \times 1.225 \)

Converting Watts/m2 to Watts

Sometimes heat generation rates for a given activity are specified in terms of W/m2, so therefore they need to be multiplied by the surface area of a human body. The average surface area of an adult human body is 1.8m2, so this may be helpful if you are using Table 1.4 from CIBSE guide A. 

Body Weight (kg) Surface Area (m²)
1 0.10
2 0.16
4 0.26
6 0.34
8 0.42
10 0.49
15 0.65
20 0.79
25 0.92
30 1.10
35 1.20
40 1.30
Table of body surface area in children, taken from nice.org.uk

Calculating CO2 generation for different age groups

To calculate CO2 generation rates for different age groups, first look up the average body weight by expected age of the occupant in the following table:

Age Range Average Weight (kg)
0-6 months 3.3 - 7.5
1-2 years 7.9 - 9.2
2-4 years 12 - 15
8-10 years 19.5 - 25.5
14-16 years 45 - 53
18-20 years 56 - 58

Next, look up the average surface area from the body weight vs surface area table above. Or, for an adult male, assume 1.8m2 as per CIBSE guide. 

Then, look up the heat generation rate per unit area of body from the following table:

Activity Heat Generation (W/m²)
Resting
Sleeping 41
Reclining 46
Seated, Quiet 58
Standing, Relaxed 70
Walking (Level)
0.9 m/s 116
1.3 m/s 151
1.8 m/s 221
Office Work
Reading, Seated 58
Writing 58
Typing 64
Filing, Seated 70
Filing, Standing 81
Lifting/Packaging 122
Occupational
Cooking 81-134
House Cleaning 99-198
Seated, Heavy Limb Movement 128
Machine Sawing 105
Light Machine Work 93-116
Heavy Machine Work 175
Handling 50 kg Bags 233
Leisure
Dancing 82-256
Tennis 210-233
Basketball 294-442
Wrestling 407-506

Figures for this table were taken from CIBSE guide A 2006, Table 1.4

Multiply the average surface area by the Heat Generation from the table above to get a metabolic rate in Watts, which you can then enter into the calculator above by selecting the Other option. 

Coincidental Loads

Understanding Load Calculations in Building Simulation: Coincidental Loads vs. Steady-State Loads

In the world of building design and simulation, accurately calculating loads is essential for ensuring optimal performance and energy efficiency. As buildings evolve to meet the demands of modern life, understanding the nuances of different load types becomes increasingly critical. Among these, coincidental loads and steady-state loads play pivotal roles in how we design systems that can adapt to varying conditions.

In Tas, loads can be calculated using Heating & Cooling Design Days via the Design Day Wizard, or directly from the Dynamic Simulation Results. In this blog post, we will explore the difference between the two methods and how this relates to coincidental and steady state loads. 

Steady-State vs Dynamic Loads

Before we discuss what coincidental loads are, we need to discuss the difference between steady-state and dynamic loads

Steady-state loads are characterised by their predictability and consistency over time. For instance, the Chartered Institution of Building Services Engineers (CIBSE) employs steady-state heating design day calculations to establish the heating requirements of a building. 

These calculations assume constant conditions for each hour of the design day, including factors such as indoor temperature, occupancy, and weather patterns. A fixed external dry bulb temperature is selected, usually -4°C in the UK, and solar , occupancy and equipment gains are removed. As the hourly conditions do not change from hour to hour, they are considered to be steady and therefore simulating this day over and over should eventually cause the results to converge. 

 

For Heating Design Days, a fixed external dry bulb temperature of -4°C is typically used for sizing in the UK. Other internal gains are removed, such as solar gain and occupancy/equipment gains.

The Design Day Wizard simplifies this process, allowing you to set up these calculations efficiently.

Similarly, the Design Day Wizard can also be utilised for cyclic cooling design day calculations, which differ slightly in that the same day of weather is simulated repeatedly but the weather does typically vary throughout the day:

 

Typically, a hot day of weather is picked for a cooling design day and this day is simulated repeatedly. Other heat gains are usually included, such as occupancy and equipment gains.

Advantages of Steady-State Loads:

  • Simplicity: Steady-state calculations are relatively straightforward and easy to implement, making them accessible for designers.
  • Consistency: They provide reliable estimates for typical operating conditions, allowing for easier compliance with regulations and standards.
  • Time Efficiency: The use of tools like the Design Day Wizard streamlines the process.

Disadvantages of Steady-State Loads:

  • Oversizing Risk: These calculations often lead to oversizing of systems since they focus on worst-case scenarios or peak demands that may not occur simultaneously. As a result, systems may be designed to handle maximum loads that are infrequent or unlikely.
  • Lack of Flexibility: Steady-state calculations do not account for fluctuations in usage or environmental conditions, which can lead to inefficiencies and higher operational costs.
  • Limited Realism: By neglecting the dynamic nature of actual building usage, steady-state calculations may not accurately reflect how systems will perform under varying conditions, potentially resulting in increased energy consumption.

In contrast, dynamic loads fluctuate in response to varying conditions and usage patterns, leading to the concept of coincidental loads. 

What are Coincidental Loads?

Imagine you are simulating a building with 5 zones. These zones are called Zone A, Zone B, Zone C etc. You simulate the entire year, and have results for every hour of the year for these zones…

One way of determining the Peak Load for these zones would be to find the maximum value in each column and sum them:

Office Zone Individual Peak Heating Load (W) Peak Occurrence Time
Zone A 10,000 Day 1, 9:00 AM
Zone B 8,000 Day 5, 9:00 AM
Zone C 12,000 Day 5, 10:00 AM
Zone D 6,000 Day 12, 11:00 AM
Zone E 5,000 Day 11, 2:00 PM
Total Individual Peak 41,000

The total individual peak loads for each office zone represent the maximum heating demand that each zone can reach. However, these peaks may not occur simultaneously; they could happen on different days and at different times.

As a result, the peak load that an HVAC system needs to accommodate when conditioning these spaces is not simply the sum of the individual peak loads. Instead, the actual peak load for the HVAC system corresponds to the specific day and hour when the combined loads of all zones are at their highest.

This is known as a coincidental load, which represents the maximum demand that the HVAC system must be prepared to meet when all zones reach their peak demand at the same time. 

Hour Zone A Heating Load (W) Zone B Heating Load (W) Zone C Heating Load (W) Zone D Heating Load (W) Zone E Heating Load (W) Total Heating Load (W)
2, 9 1,313.97 1,285.20 477.41 461.18 1,297.07 4,834.83
2, 10 730.20 665.77 245.61 212.91 715.16 2,578.65
2, 11 624.66 545.94 204.76 159.28 582.48 2,116.12
2, 12 548.08 475.79 167.82 125.19 457.60 1,774.48
2, 13 492.02 437.21 152.27 110.80 402.73 1,595.03
2, 14 427.83 406.52 144.88 114.07 384.29 1,477.59
2, 15 465.37 427.63 164.17 134.41 435.21 1,826.99

So, to be clear, the coincidental load is found by summing the rows and finding the maximum value in the Total Heating Load column throughout the year. 

In Tas, this is easily achieved in the results viewer by using the Tabular Sum table:

Coincidental Loads & Design Days

Heating Design Days produce a steady state result; the result converges, so the simulation results are the same for each hour:

As the results are the same for every hour, the coincidental peak is the same as the sum of the individual peaks due to the nature of the calculation.

With cyclic cooling design days, the results may vary hour to hour so it is possible that the coincidental load may be different to the sum of the individual peaks:

Hour Zone A (W) Zone B (W) Zone C (W) Zone D (W) Zone E (W) Coincidental (W)
193, 1 0 0 0 0 0 0
193, 2 0 0 0 0 0 0
193, 3 0 0 0 0 0 0
193, 4 0 0 0 0 0 0
193, 5 0 0 0 0 0 0
193, 6 0 0 0 0 0 0
193, 7 0 0 0 0 0 0
193, 8 0 0 0 0 0 0
193, 9 0 0 0 0 32.42 32.42
193, 10 744.19 806.96 329.64 366.47 804.12 3051.38
193, 11 817.84 877.64 359.59 395.19 896.80 3347.06
193, 12 878.05 936.04 385.12 420.38 971.08 3590.67
193, 13 917.72 976.72 403.29 440.83 1,005.45 3744.01
193, 14 975.07 1,011.81 430.39 452.72 1,047.70 3917.69
193, 15 1,026.89 1,032.86 458.94 458.89 1,083.71 4061.29
193, 16 1,067.16 1,039.12 486.24 459.08 1,107.94 4159.54
193, 17 1,099.06 1,043.97 501.24 456.24 1,128.63 4229.14
193, 18 1,020.67 995.22 434.54 423.60 1,044.74 3918.77
193, 19 0 0 0 0 0 0
193, 20 0 0 0 0 0 0
193, 21 0 0 0 0 0 0
193, 22 0 0 0 0 0 0
193, 23 0 0 0 0 0 0
193, 24 0 0 0 0 0 0

In this case, the sum of the individual peaks (4231.98W) is slightly larger than the coincidental peak (4,229.14W).  If the peak for zone D occurred at hour 17 as with the other zones, the coincidental peak would be equal to the sum of the individual peaks. 

Finding Coincidental Peaks in Tas

The method you use to obtain coincidental loads from Tas depends if you are using design days or the dynamic simulation. 

Design Days

If your design day is steady state, i.e. the results are not varying hour by hour, the coincidental load is equal to the peak load so the results from the design day wizard and the peak zone loads report can be used directly. They will be equal. 

If your design day is cyclic as is often the case with cooling design days, you can obtain the coincidental peak by looking at the tabular sum results in the TSD file and finding the maximum value.

Dynamic Simulation

If you are obtaining loads from the dynamic simulation, the Peak Loads report will report the coincidental peak in addition to the individual peaks, for the zones included in the report. 

You can also obtain the coincidental peak from the dynamic simulation by looking at the tabular sum in the TSD results and finding the peak. 

Part L 2021 Building Use Types

Part L 2021 Building Use Types

Sometimes it’s not always clear which NCM internal conditions correspond to each of the Building Use Types listed in the BRUKL document. This table lists them all out, clearly. 

 

These categories are the categories assigned by the BRUKL generator version 6.1.4.0, in conjunction with the 6.1.b internal conditions. 

Retail/Financial and Professional Services

  • A1A2_CarPark
  • A1A2_Circulation
  • A1A2_DepStoreSales
  • A1A2_DepStoreSalesChill
  • A1A2_DepStoreSalesElectric
  • A1A2_Display
  • A1A2_EatDrink
  • A1A2_FoodPrep
  • A1A2_Laund
  • A1A2_Office
  • A1A2_Plant
  • A1A2_RetWareSales
  • A1A2_RetWareSalesChill
  • A1A2_RetWareSalesElectric
  • A1A2_Sales
  • A1A2_SalesChill
  • A1A2_SalesElectric
  • A1A2_Store
  • A1A2_Toilet

Restaurants and Cafes/Drinking Establishments/Takeaways

  • A345_Circulation
  • A345_EatDrink
  • A345_FoodPrep
  • A345_Office
  • A345_Plant
  • A345_Stage
  • A345_Store
  • A345_Toilet

Offices and Workshop Businesses

  • B1_CarPark
  • B1_Changing
  • B1_Circulation
  • B1_EatDrink
  • B1_FoodPrep
  • B1_Gym
  • B1_Office
  • B1_Plant
  • B1_Reception
  • B1_Store
  • B1_Toilet
  • B1_WkshpSS

Hotels

  • C1_Bath
  • C1_Bedroom
  • C1_Changing
  • C1_Circulation
  • C1_DrySptHall
  • C1_EatDrink
  • C1_EnsuiteBedroom
  • C1_FitGym
  • C1_FoodPrep
  • C1_Laund
  • C1_Office
  • C1_Plant
  • C1_Reception
  • C1_Store
  • C1_Toilet

Residential Institutions: Residential Schools

  • C2_Schools_Bath
  • C2_Schools_Bedroom
  • C2_Schools_Changing
  • C2_Schools_Circulation
  • C2_Schools_CommunalArea
  • C2_Schools_DrySptHall
  • C2_Schools_EatDrink
  • C2_Schools_FoodPrep
  • C2_Schools_HighDensIT
  • C2_Schools_Lab
  • C2_Schools_Laund
  • C2_Schools_Lecture
  • C2_Schools_Office
  • C2_Schools_Plant
  • C2_Schools_Reception
  • C2_Schools_Store
  • C2_Schools_Teaching
  • C2_Schools_Toilet
  • C2_Schools_WkshpSS

Residential Institutions: Universities and Colleges

  • C2_Uni_Bath
  • C2_Uni_Bed
  • C2_Uni_Changing
  • C2_Uni_Circulation
  • C2_Uni_CommunalArea
  • C2_Uni_DrySptHall
  • C2_Uni_EatDrink
  • C2_Uni_FitGym
  • C2_Uni_FoodPrep
  • C2_Uni_HighDensIT
  • C2_Uni_Lab
  • C2_Uni_Laund
  • C2_Uni_Lecture
  • C2_Uni_Office
  • C2_Uni_Plant
  • C2_Uni_Reception
  • C2_Uni_Store
  • C2_Uni_Tea
  • C2_Uni_Toilet
  • C2_Uni_WkshpSS

Others: Car Parks 24 hrs

  • CarPark_CarPark
  • CarPark_Circulation
  • CarPark_Office
  • CarPark_Plant
  • CarPark_Toilet

Non-residential Institutions: Crown and County Courts

  • Court_Cell
  • Court_Circulation
  • Court_EatDrink
  • Court_FoodPrep
  • Court_Lecture
  • Court_Office
  • Court_Plant
  • Court_Reception
  • Court_Store
  • Court_Toilet

Non-residential Institutions: Education

  • D1_Edu_Changing
  • D1_Edu_Circulation
  • D1_Edu_DrySptHall
  • D1_Edu_EatDrink
  • D1_Edu_FoodPrep
  • D1_Edu_HighDensIT
  • D1_Edu_Lab
  • D1_Edu_Lecture
  • D1_Edu_Office
  • D1_Edu_Plant
  • D1_Edu_Reception
  • D1_Edu_Store
  • D1_Edu_Teaching
  • D1_Edu_Toilet
  • D1_Edu_WkshpSS

General Assembly and Leisure, Night Clubs, and Theatres

  • D2_Auditoria
  • D2_Changing_High
  • D2_Changing_Low
  • D2_Changing_Medium
  • D2_Circulation
  • D2_CirculationPub
  • D2_DrySptHall
  • D2_EatDrink
  • D2_FitGym
  • D2_FitStud
  • D2_IceRink
  • D2_Laund
  • D2_Lecture
  • D2_Office
  • D2_Plant
  • D2_Reception
  • D2_Sales
  • D2_Stage
  • D2_Store
  • D2_Toilet
  • D2_WkshpSS

Non-residential Institutions: Community/Day Centre

  • DayCtr_Changing
  • DayCtr_Circulation
  • DayCtr_DrySptHall
  • DayCtr_EatDrink
  • DayCtr_FoodPrep
  • DayCtr_Lecture
  • DayCtr_Office
  • DayCtr_Plant
  • DayCtr_Reception
  • DayCtr_Store
  • DayCtr_Toilet
  • DayCtr_WkshpSS

Residential Spaces

  • Dwell_DomBath
  • Dwell_DomCirculation
  • Dwell_DomCommonAreas
  • Dwell_DomDining
  • Dwell_DomKitchen
  • Dwell_DomLounge
  • Dwell_DomToilet

Others: Emergency Services

  • EmgcySvc_Bath
  • EmgcySvc_Bed
  • EmgcySvc_Cell
  • EmgcySvc_Circulation
  • EmgcySvc_DrySptHall
  • EmgcySvc_EatDrink
  • EmgcySvc_FitGym
  • EmgcySvc_FoodPrep
  • EmgcySvc_Office
  • EmgcySvc_Plant
  • EmgcySvc_Reception
  • EmgcySvc_Store
  • EmgcySvc_Toilet

Residential Institutions: Hospitals and Care Homes

  • Hosp_AECons
  • Hosp_Bath
  • Hosp_Bed
  • Hosp_Bed_24x7
  • Hosp_Changing
  • Hosp_Circulation
  • Hosp_CirculationPub
  • Hosp_ClassRm
  • Hosp_Diagnostic
  • Hosp_EatDrink
  • Hosp_FoodPrep
  • Hosp_IndProcess
  • Hosp_Lab
  • Hosp_Laund
  • Hosp_Lecture
  • Hosp_Office
  • Hosp_OpTheatre
  • Hosp_Physiotherapy
  • Hosp_Plant
  • Hosp_PostMortem
  • Hosp_Reception
  • Hosp_Store
  • Hosp_Toilet
  • Hosp_Wards

General Industrial and Special Industrial Groups

  • Indust_Circulation
  • Indust_EatDrink
  • Indust_FoodPrep
  • Indust_IndProcess
  • Indust_Lab
  • Indust_Office
  • Indust_Plant
  • Indust_Reception
  • Indust_Store
  • Indust_Toilet
  • Indust_WareStore

Non-residential Institutions: Libraries, Museums, and Galleries

  • LibMusGall_Circulation
  • LibMusGall_DisplayPublic
  • LibMusGall_EatDrink
  • LibMusGall_FoodPrep
  • LibMusGall_Lab
  • LibMusGall_Lecture
  • LibMusGall_Office
  • LibMusGall_Plant
  • LibMusGall_Reception
  • LibMusGall_Store
  • LibMusGall_Toilet
  • LibMusGall_WkshpSS

Others: Miscellaneous 24hr Activities

  • Misc24Hr_DataCentre
  • Misc24Hr_HeavyPlant
  • Misc24Hr_ServerRoom
  • Misc_24x7Office
  • Misc_24x7Toilet

Secure Residential Institutions

  • Prison_Bath
  • Prison_Cell
  • Prison_Changing
  • Prison_Circulation
  • Prison_ClassRm
  • Prison_DrySptHall
  • Prison_EatDrink
  • Prison_FitGym
  • Prison_FoodPrep
  • Prison_Laund
  • Prison_Lecture
  • Prison_Office
  • Prison_Plant
  • Prison_Reception
  • Prison_Store
  • Prison_Toilet
  • Prison_WkshpSS

Non-residential Institutions: Primary Health Care Building

  • PrmHlthCare_Circulation
  • PrmHlthCare_Office
  • PrmHlthCare_Plant
  • PrmHlthCare_Reception
  • PrmHlthCare_Store
  • PrmHlthCare_Toilet

Others: Stand Alone Utility Block

  • Stand_Changing_High
  • Stand_Changing_Low
  • Stand_Changing_Medium

Others: Passenger Terminals

  • Terminals_Checkin
  • Terminals_Office
  • Terminal_Circulation
  • Terminal_EatDrink
  • Terminal_FoodPrep
  • Terminal_Lounges
  • Terminal_Plant
  • Terminal_Reception
  • Terminal_Store
  • Terminal_Toilet
  • Terminal_WaitRm

Storage or Distribution

  • Ware_24x7Office
  • Ware_24x7WareStore
  • Ware_Changing
  • Ware_Circulation
  • Ware_EatDrink
  • Ware_FoodPrep
  • Ware_Office
  • Ware_Plant
  • Ware_Reception
  • Ware_Store
  • Ware_Toilet
  • Ware_WareStore

Modelling Primary and Secondary Pump Loops in TPD

We will often have to edit the water-side systems created by the wizard. One common reason is to replace a basic setup with a system which has a primary pump loop for the plant equipment, and one or more secondary pump loops distributing water around the building.

Before attempting the setups shown in this guide, you should be familiar with the topic of splitting and recombining plant flows, covered here.

Let’s begin by taking one of the setups from the previous blog post; there are three heating collections and two boilers in parallel.

The flow rates are specified rather than sized, and the correct splitting of flows is ensured by the design flow rate entered into the valves.

There is a single pump which is controlled to run when any of the collections has a load.

For this starting case, we are not using variable flow capacity on the heating collections.

We can very quickly turn this into a primary-secondary setup by replacing the valves with pumps, and adding another pipe (highlighted).

Note that the secondary pumps are connected to the controller of their respective collection. Now each secondary pump will run when its respective collection has a load. The primary pump will run whenever any collection has a load (note the MAX controller).

On hours when only one or two collections have a load, any excess flow from the primary pump will be recirculated in the primary loop.

On hours when all the collections have a load, there is no excess flow to recirculate; all of the flow from the primary pump is sent to a collection.

Note that it is possible for the sum of the secondary pump flow rates to exceed the flow rate of the primary pump; to some extent the rules established in the previous blog post do not apply for primary-secondary setups. This is potential pitfall, however, as there is no error message to warn us that the flow rates are unbalanced and we may have water flowing in an unwanted direction, leading to unwanted temperatures. In this image the flow rate of every pump is 0.9 l/s and the water temperature supplied to the collections has fallen below the target of 71 C.

If we return the pump flow rates to the correct values and delete the wires connecting the secondary pumps to their controllers, we can use the variable flow capacity option on the collections.

Let’s change how the boilers are used now. Let’s only use both boilers if the required heating load is over 50% of the maximum value.

There are a few ways we could detect this switching point; for the sake of this example we are going to test the return water temperature in the primary loop.

We have a setpoint of 71 C and a delta T of 11 C, so for a 50% load we might expect a return temperature of about 65.5 C. Let’s put a controller on the valve of one of the boilers and switch the control signal around this temperature.

We can soon see the impact on the results, compared to the case where the flow is always split evenly between the boilers.

Of course, we could just use a multiboiler component to achieve a similar result instead of two separately controlled boilers. But there may be instances where we prefer to model two separate boilers in this fashion, for example to use a wide control band as in the example above, or in a case where there are further components which we need to take into account (if, for example, one of the boilers was preceded by an exchanger to a different plant loop). There may also be a difference in pump load which needs to be taken into account; on that note, if the boilers and valves are replaced by a multiboiler we should ensure that its pressure drop matches the sum of the pressure drops of the components we have replaced (e.g. 25 kPa for the valve and 25 kPa for the boiler becomes 50 kPa for the multiboiler).

We can try to simplify the setup further; as we are using variable flow capacity on the collections we could experiment with replacing these three collections with a single one and allow the varying flow capacity to provide the flow we need (assuming we don’t need to see the heating load split between the three collections in our results). In this example the difference made to pumping power was around 0.1%, so no major error was introduced; in fact, as simpler systems generally simulate more smoothly and with fewer simulation events, in most situations we are more likely to remove a source of error by simplifying the model.

It’s not necessary with such a simple example, but it can sometimes be expedient to separate loops with a 100% efficient exchanger. This is a convenient way to stop pressure and flows from one loop affecting what’s happening in a connected loop, while still conveying 100% of the heat energy as if the water flows were uninterrupted. This is a particularly useful strategy in cases where pressure-related errors are occurring, or errors relating to negative flow through components.

Using this exchanger technique we can make a variation on an earlier setup; two boilers,
this time in series rather than parallel, with the second controlled only to be used for loads over 50% of the peak.

The first boiler in the loop is controlled to be used whenever there is a load at the collection (or rather, its pump is controlled like this, but remember the boiler will only run whenever there is flow through it).

The second boiler is controlled to be used when the water temperature leaving the first boiler is below the setpoint of 71 C, but only on hours when there is a load (note the MIN controller).

If each boiler is sized for 50% of the total load, and the exchanger efficiency is 100%, then this setup will work just fine. This could be a useful setup if we had a pump associated with each boiler and wanted to see this fact reflected in the results.

We can take this latest example and change it into a very simple-looking setup.

We have used the ancillary load property to remove the boiler pumps from the schematic and, in effect, absorb them into the multiboiler component.

We set the profile modifiers so that ancillary load 1 is used when the heating load is positive, and ancillary load 2 is used when the heating load is over 50%.

If the collection is using variable flow capacity then we don’t even require a controller for the pump.

It is often better to use an elegant solution that gives the correct results, rather than create a complicated setup which matches a schematic drawing. Whether simplification is suitable will depend on the information available and the results detail required, but there are normally several ways to reach similar or even identical results.

Splitting TPD Plant Room Flows

We will often want to create plant room layouts which are more complex than the ones created by the systems wizard, and split and combine flows to use components on separate branching pipes. There are different ways to achieve this, but also a few potential pitfalls; this guide aims to shed some light on the subject.

In the image above we have used the wizard to create three heating circuits because we wish to view the results for each collection separately. But what if we want to have the three heating collections on the same heating loop, served by the same heating source? We might decide wouldn’t be appropriate to put the collections in series as the water temperatures into the second and third collections would be too low. 

In order to put the collections in parallel we will need to use junctions to split and combine the water flow.

Each collection has a controller checking to see if the load is above zero, and the pump has a “max” controller so that it will run if any signal is greater than zero.

If we’re not careful, however, soon we could start running into problems. Here is one message we might well encounter.

Why has this happened?
Why do we have an inconsistent design flow rate?
Let’s look at the parameters which will affect flow sizing:

If we calculated the sized flow rates for the boiler and collections from their maximum loads and delta T, for this model we would find:

Boiler: 0.68 l/s
Heating A: 0.16 l/s
Heating B: 0.32 l/s
Heating C: 0.41 l/s

The 0.68 l/s of the boiler does not match the 0.89 l/s total required by the heating collections. This is the inconsistency referred to in the error message. In other words,

0.16 + 0.32 + 0.41 =/= 0.68.

How can it be fixed? We can either balance the equation or we can remove the boiler from the flow sizing altogether. If we set the boiler delta T to “none” then it won’t be used in the pump sizing and it will just accept whatever flow is required by the collections. Alternatively, we could, for example, change the delta T to 11 for the collections currently using 8; when the components have the same sizing parameters the equation will balance:

0.16 l/s + 0.23 l/s + 0.29 l/s = 0.68 l/s

This setup will still work if we want to incorporate varying flow rates; we can assign variable flow capacity to the collections and the flow rate will decrease for lower heating loads.

When we’re using the variable flow capacity on all the collections there is no more need for the controllers on the pump; the collections’ capacity will increase or decrease to allow the hot water to flow as and when it is needed, and only where it is needed, for example on this hour when only collection C has a heating load:

If we wanted to, we could take our earlier example and replace the multi-boiler with two boiler components, each with a sizing factor of 0.5, and put them in parallel; as long as the boilers are sized using the same delta T the pump sizing will succeed.

What if we didn’t want to size these flows, and instead we wanted to dictate the pump maximum flow rate and the maximum flow rate which will travel down each pipe? Let’s enter a set value for the pump flow rate and add some valves to the system.

Now we might run into an inconsistent design flow rate issue again. The error says that it might relate to the collection “Heating C”, but we shouldn’t take this too literally; this tells us that the heating system has the problem but we will have to check the whole system.

Let’s look at the design flow rate values we’ve entered into our components…

The total flow rate for the valves associated with the collections is 0.3 + 0.3 + 0.3 = 0.9, the same as the pump flow rate. So the problem probably isn’t related to the heating collections after all.

Let’s check the boilers: 0.75 + 0.75 = 1.5 l/s, which is inconsistent with the 0.9 l/s pump flow rate.

Consider what would happen where the boiler flows join before entering the pump; how could 1.5 l/s become 0.9 l/s? It cannot, so the simulation fails. On the other hand, on the other side of the pump there is no problem splitting 0.9 l/s into 0.3 + 0.3 + 0.3.

0.9 = 0.3 + 0.3 + 0.3
0.9 =/= 0.75 + 0.75

We could fix this easily by changing the boiler valve flow rates to 0.45 l/s each.

0.9 = 0.3 + 0.3 + 0.3
0.9 = 0.45 + 0.45

What if we encounter a message about an error calculating capacities?

We’ve already ensured that the flow rates are inconsistent. We now need to check either the component capacities, or pressure drops (note that only one or the other of these parameters will be specified at any time for a given component).

In this system we have specified a pressure drop for each component (and for the pump, a peak pressure value). As before, we should check the whole circuit and not just the component mentioned in the message.

Whereas with the flow rates we needed to ensure that the total flow rates on either side of a junction are equal (e.g., 0.45 and 0.45 combining into 0.9), with the pressure drops we instead need to ensure that when flows split and join the pressure drops on the branching pipes are equal to one another. Let’s break this down further:

Here we have two branches. The flow returning from the heating collections (yellow pipe) reaches a junction and is split between the two boilers. Their flows then recombine and enter the red pipe.

The pressure drop on the top branch is 25 + 50 = 75 kPa. The pressure drop on the bottom branch is 25 + 25 = 50 kPa. 50 and 75 are not equal, so we need to edit the components to balance the equation (e.g., by changing the pressure drop of the top-most valve to 25 kPa).

Meanwhile for the collections the flow from the pump (red) is split into three branches which recombine (yellow).

From top to bottom, the total pressure drop of the branches is:

25 + 25 = 50 kPa
25 + 25 = 50 kPa
50 + 25 = 75 kPa

50 and 75 are not equal. Again, we can change the pressure drop of the bottom-most valve to 25 kPa to remedy this.

We would also receive an error message like this if the peak pressure of the pump was insufficient.

In this example, if every valve, boiler, and collection had a pressure drop of 25 kPa, we could draw the diagram below:

On a complete circuit the water loses 100 kPa of pressure and the pump has to overcome this; the pump peak pressure must be 100 kPa or more.

Now the design flow rates and pressure drops are in balance, and the pump is strong enough to move the water.

The schematic is rather messy. Is there a more elegant solution?

As the error messages above have hinted, the point of the design flow rates and design pressure drops is to get a flow “capacity” value for the components. You can choose to just type this in directly.

Note that this value is not really appropriate for this component, but it is convenient; more on this later.

This approach makes it very easy to split flows equally between branching pipes.

The valves have all been deleted and a capacity of 1 has been entered into each boiler and collection.

Note that we can now adjust the ratio of flow rates between branches very easily; for example, if the capacity of the boilers is in the ratio 5:1 then the flows will take on the same ratio.

Note that this approach of using capacities will also work if we wanted to size the pump flow rate rather than specify it, as long as we set a delta T value to our collections and/or boilers.

One major downside of using capacities like this is it can be hard to get the correct pump load, which is related closely to pressure drop. By using such large capacity values, we have a negligible pressure drop and pump load. We can work around this by adding a valve in series with the pump which has a pressure drop representing the drop around the whole system, and whose flow is sized in the same way as the pump, either “Auto” (sized) or a fixed value:

As before, if it is appropriate for our design, we can turn on the variable capacity option for the collections and do away with the controllers.

In the next blog post on this topic we will look at modelling primary and secondary loops.

Drawing Dormer Windows

Windows which project from a pitched roof can be drawn without difficulty in the Tas 3D modeller. This short guide shows how to achieve this on five example models, using a methodical approach and knowledge of the “intersect planes” tool.

It is recommended that users should be familiar with different techniques for drawing roofs before following this guide. See here for more details.

Example 1

This is the easiest example, a flat-roofed wall dormer. The face of the dormer is coplanar with the external wall and there is no roof void to worry about.

Firstly, create the planes which will define the pitched roof. (On the “drawing utilities” tab, under “3D planes”, use the tool “by points”). Do not apply any planes at this stage.

Now create a plane where all points are equal to the height of the flat roof over the dormer. The reason will become clear in the next step.

Select the space. On the “drawing utilities” tab, under “3D planes”, find the tool “intersect planes”. Select the pitched roof plane and the horizontal plane which you have just created.

This will create a null wall across the space where the pitched roof and flat roof will intersect. From this line, draw null walls to the external wall (on a real project, refer to your drawings for the dormer side wall positions).

Place the window.

Recommended but optional: Clean up null lines.

Now apply the 3D planes. Note that for the flat roof over the dormer you can either use the horizontal plane created earlier or just enter a value for the space height.

Apply zones. Create the 3D analysis model.

In the list of building elements, change the null-exposed element to represent an external wall. Be sure to assign an appropriate construction to this building element in the TBD.

Finished!

In the following examples you can see how these techniques can be used to model window types that could seem challenging at first glance.

Example 2

Here we have a gable-fronted wall dormer. The face of the dormer is coplanar with the external wall and there is no roof void to worry about. But we do have to deal with two pitched roofs over the dormer – this is not a major obstacle as long as we remember how to use the intersect planes tool.

As in the previous example, begin by creating the planes which will define the main pitched roof.

Draw null lines marking the middle and limits of the dormer (on a real project, follow your drawings).

Create the planes which will define the pitched roof over the dormer. Note that it doesn’t matter if the points selected are not within the limits of the dormer’s location according to your drawings – as long as the points are on the correct line, the planes will intersect correctly with the main roof plane to define the dormer limits.

Select one of these spaces and intersect the corresponding dormer roof plane and the main roof plane. Do the same for the other space.

Place the window. Clean up null lines. Apply 3D planes.

Apply zones. Create the 3D analysis model. In the list of building elements, change the null-exposed element to represent an external wall.

Finished!

Example 3

This is similar to example 2 but we have the added complication of a roof void in some parts of the space. Thankfully, the use of 3D planes means the voids will not make our task any more difficult.

Firstly, create the planes which will define the main pitched roof. Draw null lines marking the middle and limits of the dormer (on a real project, follow your drawings).

Create the planes which will define the pitched roof over the dormer.

Select one of these spaces and intersect the corresponding dormer roof plane and the main roof plane. Do the same for the other space.

Draw the internal walls which separate the occupied space and the roof void (on a real project, follow your drawings). Also change the sides of the dormer to internal walls so that a solid wall separates the occupied space from the roof void.

Place the window. Clean up null lines. Apply 3D planes. Note that you can select multiple spaces and assign the same plane to them at the same time.

Apply zones. Create the 3D analysis model.

In the list of building elements, change the internal wall-exposed element (and if necessary, null-exposed element) to represent an external wall. Be sure to assign an appropriate construction to this building element in the TBD.

Finished!

Example 4

By now you can probably see how these techniques can be easily applied to a situation where a dormer window is set back from the external wall. Here we have another gable-roofed dormer and roof void situation, but the window is not coplanar with the external wall.

Firstly, create the planes which will define the main pitched roof. Draw null lines marking the middle and side limits of the dormer (on a real project, follow your drawings).

Create the planes which will define the pitched roof over the dormer. Note that it doesn’t matter if the points selected are not within the limits of the dormer’s location according to the plans – as long as the points are on the correct line, the planes will intersect correctly to define the dormer limits.

Select one of these spaces and intersect the corresponding dormer roof plane and the main roof plane. Do the same for the other space.

Draw the internal walls which separate the occupied space and the roof void (on a real project, follow your drawings). Change the sides of the dormer to internal walls. Draw an internal wall (not external) at the face of the dormer (on a real project, follow your drawings).

Place the window. Clean up null lines. Apply 3D planes.

Apply zones. Create the 3D analysis model. In the list of building elements, change the internal wall-exposed element (and if necessary, null-exposed element) to represent an external wall.

Finished!

Note how the different treatment of “internal wall” and “internal wall-exposed” allows us to have one building element separating the occupied space from the void space, and a different building element separating the occupied space from the outside air.

Example 5

Here we have a hip roof dormer set back from the wall, with a roof void. This is the most complicated example but by now you probably see clearly how to model it. Essentially this is the same as the previous example but with an additional step for the additional plane.

Firstly, create the planes which will define the main pitched roof. Draw null lines marking the middle and side limits of the dormer (on a real project, follow your drawings). Create the planes which will define the pitched roof over the dormer.

Draw an internal wall (not external) at the face of the dormer (on a real project, follow your drawings).

Draw a null line which intersects the dormer centre-line at the apex of the pitched roof over the front of your dormer (on a real project, follow your drawings).

Create the plane which will define the pitched roof over the front of the dormer.

Intersect planes twice. Intersect the front dormer plane with the first dormer side plane. Intersect the front dormer plane with the second dormer side plane.

Clean up null lines. This ensures nothing will interfere with intersecting the main roof plane and the dormer side planes.

Intersect the dormer side roof planes and the main roof planes.

Draw the internal walls which separate the occupied space and the roof void (on a real project, follow your drawings). Change the sides of the dormer to internal walls.

Place the window. Clean up null lines. Apply 3D planes.

Apply zones. Create the 3D analysis model. In the list of building elements, change the internal wall-exposed element (and if necessary, null-exposed element) to represent an external wall.

Finished!

Applying the techniques demonstrated in these examples will allow you to model any dormer window type you are likely to encounter.

Rhino to Tas: How to with Honeybee

Creating Tas models with Rhino & Honeybee

Rhinoceros, otherwise known as Rhino3D, is a freeform surface modeller that is gaining popularity amongst energy modellers and architects for its ease of use in creating complex geometries. Not only that, but Rhino has a diverse range of plugins which make it possible to perform building energy analysis using interoperability with simulation engines such as Tas. In this blog post, we’ll walk through using the Honeybee plugin by Ladybug tools to link Rhino and Tas. 

What are Grasshopper, Ladybug Tools & Honeybee?

Ladybug tools is a collection of applications that support environmental design founded in 2012. These applications are opensource, and are free to use. 

One of these tools is called Honeybee, and this tool is a plugin for Grasshopper, a parametric modelling tool which is part of Rhino. 

Honeybee can be used to perform daylight simulations using Radiance, and simulate energy models using OpenStudio and EnergyPlus. Honeybee can also be used to produce IDF and gbXML files, both of which provide a route into Tas. 

gbXML? IDF? What's the difference?

Both gbXML and IDF are industry standard formats that are designed to allow building design software to share and communicate data. Whilst most building design packages can export gbXML, very few produce gbXML files that contain more than just geometry.

Fortunately, IDF files often contain both geometry and the data required to perform a full building simulation. IDF files produced using Honeybee contain geometry, internal conditions, construction information, design conditions and more, so we recommend using IDF files unless you’re only interested in importing Rhino geometry into Tas. 

How do I get started?

In order to get started with Rhino and Honeybee, you’ll first need to download and install Rhino. At the time of writing, you can get an 90 day trial if you do not have a license. 

Next, you’ll need to download and install Ladybug tools from Food4Rhino. This is a zip file, so unzip it and start Rhino. Then launch Grasshopper, and open the installer.gh file within Grasshopper:

 

After running the installation script and restarting Rhino, you should see the Ladybug tools appear in Grasshopper.

Note that you will also need to install OpenStudio if you want to generate IDF files. 

Creating Rhino Geometry

Now we’re ready to create some simple geometry in Rhino, to get ready for export. Let’s start by drawing a two zone model with a couple of windows – this way we can check that our adiabatic links are created correctly, and we’ll be able to see the principles behind marking surfaces as windows. Note that there are several ways to achieve the same outcome in Rhino, so feel free to experiment!

In the video below, I demonstrate how to:

  • Set the Rhino units
  • Draw two connected cuboids
  • Draw surfaces where the windows will be

It is important to remember that two touching faces must have the same dimensions in order to form an adiabatic link when using Honeybee. 

Setting up Honeybee in Grasshopper

Now that we’ve installed Honeybee and created some geometry in Rhino, we can start setting up our Grasshopper document. In the below video, I demonstrate how to produce a gbXML and an IDF file. I also demonstrate how to modify one of the Honeybee components to produce the IDF file without simulating the OpenStudio project..which saves a lot of time!

To skip running the open studio project and copy the IDF file to a directory of our choice, we can find the line containing run_idf and comment it out using a # at the start of the line. 

				
					import shutil
        elif run_ in (1, 3):  # run the resulting idf throught EnergyPlus
            shutil.copyfile(idf,_idfPathOut)
            #sql, zsz, rdd, html, err = run_idf(idf, _epw_file, silent=silent)

    # parse the error log and report any warnings
#    if run_ in (1, 3) and err is not None:
#        err_obj = Err(err)
#        print(err_obj.file_contents)
#        for warn in err_obj.severe_errors:
#            give_warning(ghenv.Component, warn)
#        for error in err_obj.fatal_errors:
#            raise Exception(error)

				
			

We have also used shutil.copyfile to copy the IDF file to the path of our choosing using the _idfPathOut parameter we added to the ModelToOSM component. For full details, see the video above.

Commenting out the end of the script stops any error messages appearing when we attempt to run the component. 

Importing the IDF into Tas

Once we’ve created our IDF file, we can import it into Tas using the IDF tool. We can optionally create a Tas3D model for shading calculations, import constructions & geometry and even specify the weather we want to use to simulate. In other words, the model is ready to go!

Rhino to Tas Workflow Summary

Below is a labelled Grasshopper diagram showing the key steps in creating gbXML and IDF files from Rhino. Remember, this is just one way to do it! Some of the components can be linked together in a different order, and some can be omitted. If you wanted to model % glazing, for example, you could add the apertures straight to the HoneyBee model and skip specifying and combining them. 

As the Rhino files are saved separately to the Grasshopper files, you can save the Grasshopper files and re-use them on future projects just by updating the geometry components each time. 

What else is Possible with Tas & Rhino?

As Grasshopper provides a visual programming interface to Rhino and Tas has a programming interface, the two can be linked to achieve endless outcomes – from parametric runs to performing full energy simulations and visualising the results in Rhino. 

You can create your own Tas Grasshopper code modules via python scripts:

				
					import win32com.client

#open the building simulator and then open a file
tbdDoc = win32com.client.Dispatch("TBD.Document")
tbdDoc.openReadOnly("C:\\Users\\hilmya\\Desktop\\tas house.tbd")

i = 0
while(tbdDoc.Building.getZone(i) is not None):
    #print each zone name to console
    print(tbdDoc.Building.getZone(i).name)
    i = i+1
				
			

Better yet, you can create C# modules in Grasshopper which have the advantage of code completion:

In order to use the Tas automation interface with a c# script component in Grasshopper, right click on the component and click Manage Assemblies. Then reference the appropriate dll, for example, for TBD you would reference Interop.TBD.dll from the Tas installation directory. 

You can do the same with the Visual basic (VB script) component. 

Ben Abel shares how Hilson Moran use the Tas API

Ben Abel shares how Hilson Moran use the Tas API

Ben Abel is a Director and Head of Research and Development at Hilson Moran, and we recently spoke to him to learn how Hilson Moran use the Tas Application Programming Interface to be one of the most advanced cutting edge designers of the built environment.

Hilson Moran have been using Tas for 25 years

HM has been a long-time user (25 years) of advanced computer modelling techniques in the built environment. EDSL TAS has been the preferred dynamic thermal modelling (DTM) tool of choice over that period and has been used on many iconic and significant projects in the business from 30 St. Mary Axe (The Gherkin) in London, to three of the stadia for the Qatar 2022 World Cup and many others in-between.

Noticing Trends: Coding

The recent upturn and interest in coding by users to create custom interfaces has become increasingly important to improve productivity and allow the outputs to meet the end user/client requirements.

TAS was an early adopter of allowing access to the underlying functions in the software through the Application Programming Interface (API). HM has used this feature extensively to create a range of tools to improve both functionality and productivity.

The types of tools range from bespoke interfaces, for areas such as thermal comfort, HVAC plant modelling and dynamic façade control, and interoperability in using the software through applications such as Grasshopper to allow the DTM information to interact with other outputs from a variety of software.

TM59 Parametric Tool & the CIBSE Building Simulation Group Awards

The HM creation of a TM59 parametric tool, using the TAS API, resulted in the tool being a CIBSE Building Simulation Group Awards finalist at the event at Build2Perform in November 2022.

The tool was used to test and inform the values which were used in the simplified method of the new Building Regulation Part O. The tool allowed hundreds of models to be tested against the overheating criteria to ensure the glazed and ventilations areas presented in the simplified method were in line with the full dynamic methodology.

This task would have been excessively time consuming to achieve manually and at risk of human error.

HM will continue to explore and develop the use of TAS through the API as the benefits it brings are immense due to the efficiencies of automation and the wider sharing and integration of data.

Drawing Roofs

There are many options for drawing roofs in the Tas3D modeller. This short guide explains when each option may be the best choice and clears up some common questions.

Option 1: Set Wall Height
Pros: A very quick solution for simple sloped roofs with a level ridge.
Cons: Unsuitable for any other type of roof.

Option 2: Set Space Height
Pros: A very quick solution for stepped flat roofs.
Cons: Unsuitable for sloped roofs.

Option 3: Set Point Height (Select Join)
Pros: Allows quick modelling of curved roofs and intersecting slopes.
Cons: Can be time-consuming with roofs that cannot be triangulated easily. Not suitable when there is an abrupt change in roof level.

Option 4: Use 3D Planes
Pros: Can be applied to any sloping roof situation. Plane can be used for multiple roof areas at once. Intersection line between two planes can be calculated automatically. Reduces risk of errors arising from incorrect wall and point heights.
Cons: Creating the planes can be more time-consuming than the other methods.

Examples

How would three different roofs be modelled most effectively on this simple building?

With this roof there is a sudden change in roof level, and the “Set Point Height” option cannot be used. We can see why if we consider one of the points used by both sloping roofs (highlighted in lower image) which would need to have two different roof heights at the same time; in this example it would need to be 4.5m for the left-hand roof and 5.5m for the right-hand roof.

We need to use 3D Planes here.

With this example we have a sudden change in roof level, meaning that once again we have points which would need to have two different heights at once; we cannot use the “Set Point Height” option.

In this case we would need to use 3D Planes for the left-hand roof. For the right-hand roof we can simply use the “Set Space Height” option.

This roof rises to a single point and there is no step or sudden change in roof level. The roof can be achieved easily by using the “Set Point Height” option.

What about a situation where the roof itself is very simple, but there are several internal walls underneath it?

The answer depends on whether or not the roof is on a separate storey. In the case where the internal walls extend upwards to meet the underside of the sloping roof, the best option is to use 3D Planes. But if there is a separate roof space and the internal walls only extend to the underside of a flat ceiling, we should model the roof on a separate storey and the “Set Point Height Option” can be used.

What about “gaps” in the roof where internal walls or null lines are exposed?

When you create the analysis model, Tas3D creates a new building element for the exposed parts of internal walls, null walls, etc. These new elements, which will have a name ending in “-exposed” can be changed by the user to represent external walls (or, depending on your building, you may want to set these up to represent, e.g., glazing). When you refresh the analysis model you will see that your roof “gaps” have been filled. Be sure to assign an appropriate construction to these building elements in the TBD.