Fabrication Part - Show Service
						
					
					
				
			
		
	
			
	
	
	
	
	
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report
This post started as a question, but I figured it out and thought someone might be interested. I couldn't find any examples of this in the API samples or through web searches, and ChatGPT failed miserably.
My goal is to replicate the Show Service button that appears in the Modify tab when you select a Fabrication Part.
Clicking this button opens the MEP Fabrication Parts panel to the matching service and palette that generated that part, and highlights the button that was used.
I snooped through the attributes and parameters of several fabrication parts but could only find ServiceId indicating what button was used to draw the part. The fabrication button name does not necessarily correspond to the name of the part drawn in Revit.
Example: I have a button named "Pipe":
These are the relevant properties of the button object, which I obtained like this:
# doc is the Autodesk.Revit.DB.Document in question
# service_id, palette_index, and button_index are given integers
fabrication_configuration = FabricationConfiguration.GetFabricationConfiguration(doc)
service = fabrication_configuration.GetService(service_id)
button = service.GetButton(palette_index, button_index)
print(button.ButtonIndex) # == 0
print(button.Code) # == P01
print(button.ConditionCount) # == 1
print(button.IsAHanger) # == False
print(button.IsStraight) # == True
print(button.Name) # == Pipe
print(button.PaletteIndex) # == 0
print(button.ServiceId) # == 3381
Not much information about what will be drawn in Revit. There is a method FabricationServiceButton.ContainsFabricationPartType(partType) that I finally found to be the key.
The part drawn in Revit looks like this ("Cerro Type L Hard Copper Pipe"):
It has attributes and properties like this:
# select your part
ref = uidoc.Selection.PickObject(ObjectType.Element)
element = doc.GetElement(ref)
# properties
print(element.ProductName) # == Pipe
print(element.ProductLongDescription) # == Type L Hard Copper Pipe
print(element.ProductOriginalEquipmentManufacture) # == Cerro
print(element.ProductSpecificationDescription) # == Type L
print(element.Name) # == Default
# parameters
print(element.LookupParameter('Product Name').AsValueString()) # == Pipe, also BuiltInParameter.FABRICATION_PRODUCT_DATA_PRODUCT
print(element.LookupParameter('Product Long Description').AsValueString()) # == Type L Hard Copper Pipe, also BuiltInParameter.FABRICATION_PRODUCT_DATA_LONG_DESCRIPTION
print(element.LookupParameter('OEM').AsValueString()) # == Cerro, also BuiltInParameter.FABRICATION_PRODUCT_DATA_OEM
print(element.LookupParameter('Product Specification Description').AsValueString()) # == Type L, also BuiltInParameter.FABRICATION_PRODUCT_DATA_SPECIFICATION
Then I discovered the FabricationPartType class, which you can get from a fabrication part's GetTypeId() method like this: doc.GetElement(element.GetTypeId()). The API documentation says "For the product-based MAP parts, every size is a new part type in Revit. For others, one part type can have many sizes."
Now I can put FabricationServiceButton.ContainsFabricationPartType(partType) together with element.GetTypeId(). There is one last step, which took me a little longer. Sometimes we have several ITM files in a single button:
Revit classifies each of these as a Condition. ContainsFabricationPartType() only tells you that the button contains the given partType; now you have to get the condition that matches the partType. For that you have to go back to the FabricationPartType class and use the Lookup(document, button, condition) method with each valid condition the button contains.
This is the summary.
# you can run this in RevitPythonShell
fabrication_configuration = FabricationConfiguration.GetFabricationConfiguration(doc)
# select a FabricationPart
# -- consider creating an ISelectionFilter that only passes elements of type FabricationPart
ref = uidoc.Selection.PickObject(ObjectType.Element)
element = doc.GetElement(ref)
part_type = doc.GetElement(element.GetTypeId())
service_id = element.ServiceId
service = fabrication_configuration.GetService(service_id)
matching_button = None
matching_condition = None
for palette_index in range(service.PaletteCount):
    for button_index in range(service.GetButtonCount(palette_index)):
        button = service.GetButton(palette_index, button_index)
        for condition in range(button.ConditionCount):
            part_type_element = FabricationPartType.Lookup(doc, button, condition)
            # the GUI always picks the first button and condition that matches -- the efficiency of this would be improved by making this into a a function that yields the first match
            if part_type_element != ElementId.InvalidElementId and matching_button is None:
                matching_button = button
                matching_condition = condition
I think you can now use this matching button, for example, to replicate Create Similar command for fabrication parts, using the second overload with matching_button and matching_condition from above.
public static FabricationPart Create(
	Document document,
	FabricationServiceButton button,
	int condition,
	ElementId levelId
)
 
     
             
		
			 
					
				
		
 
		
			 Developer Advocacy and Support +
 Developer Advocacy and Support + 