8. Revit API UI: DockablePane(可停靠窗口)
接着前面的预览控件和外部事件,这一篇,我们来看看可停靠窗口
DockablePane
的实现。
实现流程
可停靠窗口相关的类与接口主要有
DockablePane
和
IDockablePageProvider
,都是在
UI
命名空间下。
流程:
- 创建一个
Page
页,并实现IDockablePageProvider
接口; - 生成一个
DockablePaneId
,需要传入一个Guid
,所以最好建立一个映射表; - 在插件入口(即
IExteranlApplication
)中注册该窗口,使用UIControlledApplication.RegisterDockablePane(..)
方法; - 使用时,通过
uiApp.GetDockablePane(dpId)
方法检索到可停靠窗体,调用Show()
显示窗体。
先实现,后面再讲讲我个人测试的遇到的问题。
示例
本例想接着用上一篇用到的预览窗体界面,但Revit启动时显然是没有
View
的,就算了。
XAML
:新建一个叫"DockablePage"的Page
,里头随便放些东西。记得要实现IDockablePageProvider
接口。IExternalApplication
:在外部应用启动时注册。
[Transaction(TransactionMode.Manual)][Journaling(JournalingMode.NoCommandData)]
public class Application : IExternalApplication
{
public Result OnStartup(UIControlledApplication application){
this.RegisterDockablePane(application);return Result.Succeeded;}
public Result OnShutdown(UIControlledApplication application){return Result.Succeeded;}
private voidRegisterDockablePane(UIControlledApplication application){
DockablePaneId dockablePaneId = new DockablePaneId(new Guid(Mapping.DockablePaneId));// 生成ID
application.RegisterDockablePane(dockablePaneId,"DockablePane", DockablePage.GetDockablePage());// 进行注册}}
IExternalCommand
:在外部命令中使用。
[Transaction(TransactionMode.Manual)][Journaling(JournalingMode.NoCommandData)]
internal class DockablePaneCommand : IExternalCommand
{
public Result Execute(ExternalCommandData commandData, ref string message, ElementSet elements){
UIApplication uiApp = commandData.Application;
DockablePaneId dockablePaneId = new DockablePaneId(new Guid(Mapping.DockablePaneId));// 创建ID,根据ID查找//TaskDialog.Show("DockablePaneCommand", $"{DockablePane.PaneIsRegistered(dockablePaneId)}\r\n" + $"{DockablePane.PaneExists(dockablePaneId)}");
DockablePane dockablePane = uiApp.GetDockablePane(dockablePaneId);
dockablePane.Show();return Result.Succeeded;}}
至此,一个完整的可停靠窗口的Demo就完成了。
记录问题
1. 必须在外部应用中注册吗
如果你了解
Application
相关API,就能够知道
UIControlledApplication
和
UIApplication
的大部分类成员都是一样的,效果一样。不了解没关系可以看这篇:5. Revit API: Application。
那么,就一定得在外部应用中注册?
是的,必须在外部应用中注册。
能不能在外部命令中,通过UIApplication注册呢?
答案是可以的,但是只能注册,无法显示。
DockablePane
类中有3个静态方法,通过传入
DockablePaneId
进行检测。
方法描述
PaneIsBuiltIn(dpID)
检测目标窗口是否为Revit内置窗口
PaneIsRegistered(dpID)
检测目标窗口是否已注册
PaneExists(dpID)
检测目标窗口是否已创建
通过这三个方法,可以得到结果:
注册位置内置已注册已创建IExternalApplication❌✔✔IExternalCommand❌✔❌
当在外部命令中注册,并调用
Show()
方法时,会报错。
2. 必须用Page吗
虽然API文档中写建议使用Page及其派生类。那我可偏要看看不用能咋样。
好吧,不用界面就黑了。
Window
不行,
Page
和
UserControl
都可以。Window是最高级别的UI元素,猜测非Window都可以。
下面是使用Window和UserControl的截图
3. 可停靠窗口中,可以放外部事件吗
可以
上面说了,最开始是想直接使用上一篇的界面,那上面就有
ExternalEvent
,我测试了那个删除按钮还是有效的。
4. 如何让可停靠窗口的内容变化
例子中创建了个表格,就是想从项目文档中获取信息,进行展示。就像Revit自带的那些窗口一样,内容可以变化,想想就很棒是不是,可惜不得行。
不知道,不会,没搞定😶
好吧,没搞定。
DockablePane
类中,并没有获取窗体的方法或属性,于是尝试曲线修改。
- 尝试让数据表绑定在一个静态变量上,可是数据变了没效果。
- 尝试将页面设为单例,改了还是无效。
然后就没有然后了,怀疑注册时传进去的和Revit创建的就不是同一个东西。
5. 遇到的其它问题
遇到了一个"无法初始化附加模块xxx,因为“…/xx.dll”不存在"的问题。
显然这破问题的原因不在于那个dll不存在。
问题在哪儿呢?还是之前写
Ribbon
时偷懒了,直接用"xxx"代替了。
这里问题的原因是,Ribbon控件里,命令的路径错了
Ribbon那篇没有示例图片,这里补充一个吧。
private voidCreateRibbon(UIControlledApplication application){
string tabName ="Samples";
string panelName ="Commands";
RibbonPanel panel = application.CreatePanel(panelName, tabName);
PushButtonData buttonData_single = new PushButtonData("单个按钮","单个按钮", addInPath,"RevitAPISamples.Commands.Dialog"){
LargeImage = new BitmapImage(new Uri(Path.Combine(this.buttonIconsFolder,"CreateWall.png"))),
Image = new BitmapImage(new Uri(Path.Combine(this.buttonIconsFolder,"CreateWall.png"))),
ToolTip ="按钮(单个)",
LongDescription ="鼠标放按钮上,就会出现这段文字",
ToolTipImage = new BitmapImage(new Uri(Path.Combine(this.buttonIconsFolder,"Revit.png"))),};
ContextualHelp contextualHelp = new ContextualHelp(ContextualHelpType.Url,"https://thebuildingcoder.typepad.com/blog/2020/09/on-spaces-in-help-and-renaming-a-parameter.html");
buttonData_single.SetContextualHelp(contextualHelp);
panel.AddItem(buttonData_single);}
总结
UI
篇还有多少来着,不记得了,得去翻翻之前的图。
这篇就到这里了,下篇写什么呢,emm…。
版权归原作者 杂鱼Tong 所有, 如有侵权,请联系我们删除。