Using Excel in Automations

Let's say you want your non-coding coworkers to be able to control certain results of your scripts. There are several ways to do this, but by far the simplest is to create a shared Excel file. Your coworkers can input values into this file, which can then be read by your script at execution time. There are countless possible use-cases for this method, but for this example, we'll keep it simple and look at creating levels at certain elevations.

Setting Up Excel

Before we create any code, we need to set up our Excel file. Make sure to save it somewhere your coworkers can access. We'll create two columns, one called "LevelName" and the other "Elevation". We'll call our sheet "New Level Names".

NOTE: Once the file is created, we'll want to add guards to make sure our script runs smoothly. We don't want people to be able to add non-integers to the Elevation column, for instance. For more information on protecting Excel files from faulty entries, click here.

Setting Up the Script

With that sorted, we'll turn to our code. The first thing our code needs to do is read our Excel file and convert the information it finds into a python dictionary. There's a few libraries we could use here, but I'll stick to pandas. In the example below, we use read_excel() to create a dataframe ("df"). We then convert our dataframe to a dictionary. To do this, we then use set_index() to tell pandas that we want to use "LevelName" column as our index, so its values will become the keys of our dictionary. We set the "Elevation" column as our values column. We then use to_dict() to actually create the dictionary.
doc = DocumentManager.Instance.CurrentDBDocument
target_path = fr"lorem_ipsum"
# Create dataframe from excel path
df = pandas.read_excel(target_path, sheet_name = "New Level Names", header = 0, dtype = str)
# Convert the DataFrame to a dictionary
df_dict = df.set_index("LevelName")["Elevation"].astype(int).to_dict()
We should now have a dictionary called df_dict that looks something like this:
{
"Apple":1,
"Banana":2,
"Cherry":3
}
We can now iterate through this dictionary to create our levels. Before we do this, we'll collect all our existing levels so we can skip any that already exist.
existing_levels = FilteredElementCollector(doc).OfClass(Level).ToElements()
existing_level_names = [level.Name for level in existing_levels]
TransactionManager.Instance.EnsureInTransaction(doc)
for levelname, elevation in df_dict.items():
if levelname not in existing_level_names:
new_level = Level.Create(doc, elevation)
new_level.Name = levelname
TransactionManager.Instance.TransactionTaskDone()
And that's it! Of course, we can beef this code up. For instance, we can update levels' elevations if these values are changed in our Excel sheet. To do this, we'll change the way we structure existing_level_names to make switching between our layer_names and layer_objects a little easier:
existing_levels = FilteredElementCollector(doc).OfClass(Level).ToElements()
existing_level_names = {}
for level_object in existing_levels:
level_name = level_object.Name
existing_level_names[level_name] = level_object
TransactionManager.Instance.EnsureInTransaction(doc)
for levelname, target_elevation in df_dict.items():
if levelname in existing_level_names:
target_level = existing_level_names[levelname]
elevation_parameter = target_level.get_Parameter(BuiltInParameter.LEVEL_ELEV)
current_elevation = elevation_parameter.AsDouble()
if current_elevation != target_elevation:
elevation_parameter.Set(target_elevation)
TransactionManager.Instance.TransactionTaskDone()
There's a few more things we can do, such as adding functionality for deleting levels, but it all works on the same principle.

If people in our team add or change values in our excel sheet, we'll see those changes after the next execution of our script without having to change a thing ourselves.

Comments

Popular posts from this blog

Check which user owns an element in a shared model

Create dimensions

Create Dimensions - Rooms