Here is an issue that has come up a few times now and is worthwhile clarifying: how to operate on a Revit document from a dockable panel.
The dockable panel UI API framework was introduced in Revit 2014. Its use is demonstrated by the DockableDialogs SDK sample, including modeless dialog design and external events, and also by the simpler dockable panel sample.
The fact that a dockable panel lives in a modeless context is apparently confusing for some. Since the panel is active at all times, you are not inside a Revit external command handler and therefore not in a valid Revit API context.
This leads to queries like this:
Question 1: I'm probably overlooking something, but how do you call an IExternalCommand from a dockable window. I guess this is possible, right?
Answer: No, sorry. See below.
Question 2: I create my own dockable dialog. It loads a table from an external data source and displays that information in a tree view. That works fine.
Now I want to add a button to that form to retrieve a selected Revit element and attach that to one of the tree nodes (cf. discussion group comment). In fact, I just want to keep it simple and copy some tree node data to a Revit shared parameter.
I implemented the code to determine the right tree node.
However, how can I get this button to connect to the current active Revit document? Am I doing something wrong here? Is that possible with this at all?
I noticed an outdated AUGI discussion thread on accessing a Revit document from a Windows form that seems to implicate that I need to pass the Application object to the XAML form. However, the explanation in that answer is not enough for me to solve my problem.
Now I also discovered that starting a transaction from an external application running outside of a valid Revit API context is not allowed.
How can I solve this, please?
Answer: As mentioned in the introduction above, a dockable panel is similar to a modeless form, which is similar to an external Windows application, which is basically the same as any other process outside of Revit: it has not been registered with the Revit API, Revit is completely unaware of its existence, and you can make no use whatsoever of the Revit API from that process.
Modeless access to the Revit API boils down to accessing the Revit application from an arbitrary external application.
The Revit API does not provide any direct access to the Revit application, except in the valid Revit API context situations defined by the various Revit API events. Basically, the complete Revit API is based entirely on call-back methods. There is no API support for driving Revit from an external application, whether .NET, WPF, or anything else.Being in a modeless context, you have no valid Revit API context. You can subscribe to the Idling event to obtain one, and the available dockable dialogue samples demonstrate how to do so.
Workarounds exist, have been frequently described, both here and elsewhere, and numerous previous discussions address various aspects of this situation.
The nicest and most recent complete sample I know of is my cloud-based round-trip 2D Revit model editor project.
Here are some other recent mentions of the topic:
- Display real-time interactive element properties
- Behind the scenes of the NBS Revit add-in
- Batch mode data extraction
For more background information, please look at the following blog categories and keyword search results:
Happy exploring!