3 自己的工具按钮

上次的例子只能在“附加模块”→“外部工具”下运行,用作个人作品是没问题,如果打算搞个公司产品的话,估计BOSS是不会满意的。这次我来做一个直接显示在“附加模块”选项卡上的工具按钮。

3.1 基础

1、新建一个项目WelcomeToRevit。添加引用RevitAPI.dll和RevitAPI.dll。具体做法和上次一样,不赘述。

2、保留指令
using System;
添加指令
using Autodesk.Revit.DB;
using Autodesk.Revit.UI;
using Autodesk.Revit.Attributes;

3、把默认生成的public class Class1{}改成
public class Class1: IExternalApplication{}

4、在public class Class1: IExternalApplication{}内部添加两个函数OnStartup(UIControlledApplication application){}和OnShutdown(UIControlledApplication application){},分别用Result加以约束,并设定返回值为Succeeded。

5、在OnStartup(UIControlledApplication application){}内添加以下内容:
RibbonPanel ribbonPanel = application.CreateRibbonPanel("演示");
PushButtonData buttonData = new PushButtonData("cmdWelcomeToRevit","欢迎",@"C:\RevitBlog\Projects\WelcomeToRevit\WelcomeToRevit\bin\Debug\WelcomeToRevit.dll","WelcomeToRevit.Class2");
PushButton pushButton = ribbonPanel.AddItem(buttonData) as PushButton;
pushButton.ToolTip = "欢迎来到Revit。";

6、在命名空间WelcomeToRevit内部(不是Class1内部)添加一个类Class2,继承于IExternalCommand接口类。在该类前面添加[Transaction(TransactionMode.ReadOnly)],在入口函数里添加内容TaskDialog.Show("RevitDemo", "Welcome To Revit!");并返回Succeeded。

7、完成后的代码如下:

 1 using System;
2 using Autodesk.Revit.DB;
3 using Autodesk.Revit.UI;
4 using Autodesk.Revit.Attributes;
5
6 namespace WelcomeToRevit
7 {
8 public class Class1: IExternalApplication
9 {
10 public Result OnStartup(UIControlledApplication application)
11 {
12 RibbonPanel ribbonPanel = application.CreateRibbonPanel("演示");
13
14 PushButtonData buttonData = new PushButtonData("cmdWelcomeToRevit","欢迎",@"C:\Test\WelcomeToRevit\WelcomeToRevit\bin\Debug\WelcomeToRevit.dll","WelcomeToRevit.Class2");
15 PushButton pushButton = ribbonPanel.AddItem(buttonData) as PushButton;
16
17 pushButton.ToolTip = "欢迎来到Revit。";
18
19 return Result.Succeeded;
20 }
21 public Result OnShutdown(UIControlledApplication application)
22 {
23 return Result.Succeeded;
24 }
25 }
26
27 [Transaction(TransactionMode.ReadOnly)]
28 public class Class2 : IExternalCommand
29 {
30 public Result Execute(ExternalCommandData commandData, ref string message, ElementSet elements)
31 {
32 TaskDialog.Show("RevitDemo", "Welcome To Revit!");
33 return Result.Succeeded;
34 }
35 }
36 }

仔细对比一下上个例子,会发现只是多了从8行到25行这一段。
8行,声明一个类,继承自RevitAPI的IExternalApplication(外部程序)接口。与IExternalCommand不同,IExternalApplication用两个成员函数OnStartup和OnShutdown来对应启动和关闭状态。
10行,重载OnStartup函数。
12行,实例化对象RibbonPanel,并通过CreateRibbonPanel方法新建一个面板,这个面板是用来承载按钮的,面板下方显示“演示”字样。
14行,实例化对象PushButtonData,并通过构造函数新建一个按钮数据。该构造函数有四个参数:第一个是程序内部使用的按钮名称,这个只要注意不要重名就可以;第二个是用户可见的按钮文字,这个可以自己随便写;第三个是按钮对应的类库地址,即dll的路径,要按实际的填;第四个是按钮命令实现代码的类名,这里要和28行对应。其中第三个参数前面的@用来忽略后面的转义字符“\”,这是个很实用的小技巧。
15行,根据按钮数据实例化对象PushButton,并把这个按钮添加到面板上。
17行,给按钮添加一个提示,当鼠标停在上面时会出现相应提示。

8、编写WelcomeToRevit.addin复制到revit插件目录下。

 1 <?xml version ="1.0" encoding ="utf-8" standalone ="no"?>
2 <RevitAddIns>
3 <AddIn Type="Application">
4 <Assembly>
5 C:\test\WelcomeToRevit\WelcomeToRevit\bin\Debug\WelcomeToRevit.dll
6 </Assembly>
7 <AddInId>86394068-436d-4d15-ba53-96eefd77252c</AddInId>
8 <Name>WelcomeToRevit</Name>
9 <FullClassName>WelcomeToRevit.Class1</FullClassName>
10 <VendorId>NAME</VendorId>
11 </AddIn>
12 </RevitAddIns>

对比上个例子,有几点不同:
<AddIn Type>标签由"Command"变成"Application",表明本插件是外部应用而不是外部命令。
没有<Text>标签,这个标签对外部应用不起作用。
增加<Name>标签,为插件指定一个名字,这个标签只用于外部应用。

3.2 美化

测试一下刚完成的插件,这次在“附加模块”里出现了独立的按钮,执行后弹出对话框,不过没有图片,看上去不够美观。接着我来个它添加图片。

1、在本插件所在目录下建立一个文件夹,丢一个图片文件进去。

2、在项目中添加引用WindowsBase、PresentationCore和System.Xaml(都是框架)。

3、添加指令using System.Windows.Media.Imaging;

4、在OnStartup函数里添加如下内容:
Uri uri = new Uri(@"C:\test\WelcomeToRevit\WelcomeToRevit\bin\Debug\images\wtr.png");
BitmapImage bitmapImage = new BitmapImage(uri);
pushButton.LargeImage = bitmapImage;
也可以直接写成
pushButton.LargeImage = new BitmapImage(new Uri(@"C:\test\WelcomeToRevit\WelcomeToRevit\bin\Debug\images\wtr.png"));

重新编译生成,执行后看到按钮有图片了。

简单解释一下:
Autodesk.Revit.UI的PushButton有个参数LargeImage用来定义按钮图片,但这个参数的类型是BitmapImage,不能直接接收Image类型,所以要通过Uri来转换。

3.3 进阶

其实上面的代码还是有缺陷的,就是把插件和按钮图片的路径在程序里写死了,这样很不灵活。所以我还要改一下。

1、添加指令using System.Reflection;

2、在定义按钮数据的语句前加上string thisAssemblyPath = Assembly.GetExecutingAssembly().Location;用以获取程序路径。

3、把定义按钮数据的语句改成PushButtonData buttonData = new PushButtonData("cmdWelcomeToRevit","欢迎",thisAssemblyPath,"WelcomeToRevit.Class2");就是替换掉第三个参数。

插件路径的问题解决了,再来看按钮图片的问题。小火车公布的源码用的是Replace方法,用图片文件名替换掉dll文件名,但它把dll文件名写死了,所以还是不够灵活,所以我用另一个方法。

4、添加指令using System.IO;

5、在定义Uri的语句前加上string imagePath = Path.GetDirectoryName(thisAssemblyPath) + @"\images\wtr.png";

6、把定义Uri的语句改成Uri uri = new Uri(imagePath, UriKind.RelativeOrAbsolute);

最终代码如下:

 1 using System;
2 using System.IO;
3 using System.Reflection;
4 using System.Windows.Media.Imaging;
5 using Autodesk.Revit.DB;
6 using Autodesk.Revit.UI;
7 using Autodesk.Revit.Attributes;
8
9 namespace WelcomeToRevit
10 {
11 public class Class1: IExternalApplication
12 {
13 public Result OnStartup(UIControlledApplication application)
14 {
15 RibbonPanel ribbonPanel = application.CreateRibbonPanel("演示");
16
17 string thisAssemblyPath = Assembly.GetExecutingAssembly().Location;
18 PushButtonData buttonData = new PushButtonData("cmdWelcomeToRevit","欢迎",thisAssemblyPath,"WelcomeToRevit.Class2");
19 PushButton pushButton = ribbonPanel.AddItem(buttonData) as PushButton;
20
21 pushButton.ToolTip = "欢迎来到Revit。";
22
23 string imagePath = Path.GetDirectoryName(thisAssemblyPath) + @"\images\wtr.png";
24
25 Uri uri = new Uri(imagePath, UriKind.RelativeOrAbsolute);
26 BitmapImage bitmapImage = new BitmapImage(uri);
27 pushButton.LargeImage = bitmapImage;
28
29 return Result.Succeeded;
30 }
31 public Result OnShutdown(UIControlledApplication application)
32 {
33 return Result.Succeeded;
34 }
35 }
36
37 [Transaction(TransactionMode.ReadOnly)]
38 public class Class2 : IExternalCommand
39 {
40 public Result Execute(ExternalCommandData commandData, ref string message, ElementSet elements)
41 {
42 TaskDialog.Show("RevitDemo", "Welcome To Revit!");
43 return Result.Succeeded;
44 }
45 }
46 }

3.4 总结

本例主要有如下知识点:

1、外部应用类型插件的基本写法和相应配置文件的基本写法。

2、在“附加模块”选项卡里添加按钮。

3、给按钮添加图片。

4、使用相对路径。

踏上Revit二次开发之路 3 自己的工具按钮的更多相关文章

  1. 踏上Revit二次开发之路 1 准备工作

    1 准备工作 工欲善其事,必先利其器.在正式开始之前,我觉得有必要先盘点一下需要准备些什么. 1.1 硬件设备 PC机一台(谢绝Apple). 配置不能太低,至少要i3以上的cpu.4g以上的内存和支 ...

  2. 踏上Revit二次开发之路 0 序

    0 序 近来,由于工作上的需要,开始自学Revit二次开发. Revit由欧特克公司专为BIM构建,是建筑业体系中使用最广泛的软件之一.借助欧特克公司在我国市场占有率方面的绝对优势,甚至给不少人带来& ...

  3. 踏上Revit二次开发之路 2 从“HelloWorld”入手

    2 从"HelloWorld"入手 在欧特克的官方网页上有个叫<My First Plug-in Training>的项目,号称可以让一个完全没有编程基础的人照着做出一 ...

  4. Revit二次开发之绘制钢筋

    第一次在博客园上写东西,也不知道该写些什么,我想就写点最近项目到遇到的问题吧. 最近在做一个小项目,具体需求大概是在一个revit模型中的对应的楼板位置绘制钢筋. 由于刚接触Revit二次开发,之前也 ...

  5. Revit二次开发初体验

    最近换了下工作,由之前的互联网企业转入了BIM软件开发行列.具体原因不多说,作为一个程序员来说学习永无止境.下面来一个Hello World体验下Revit的二次开发 事前准备 VS Revit 20 ...

  6. Revit二次开发 推荐

    学习revit二次开发,建议还是先把revit熟悉一下,去建立一下模型,亲自感受一下是如何创建模型的流程,其中会遇到什么问题.这样在自己做二次开发的时候,一些问题自己就能提前想到,规避掉.我大概用了半 ...

  7. [Revit]Autodesk Revit 二次开发整理(资料、准备工作和环境搭建)

    1 前言 Revit被Autodesk收购之后,整理和开放了一大部分API,供开发者实现自己的功能和程序,总体来说API的功能比较完善,毕竟市面上已经出现了各式各样的插件. 本人也是初学者,在Revi ...

  8. Revit二次开发环境配置(Revit 2020 +Visual Studio 2019)

    Revit二次开发环境搭建(Revit 2019+Visual Studio 2017)准备内容 Revit 2019开发环境的搭建,需要安装的内容如下: Revit 2019(主要的开发环境) Vi ...

  9. Revit二次开发之创建风管

      在Revit中,风管用于连接管件,风道末端和机械设备,今天简单尝试了下使用RevitAPI创建风管,现分享下我的方法.   风管从类型上可分为三类:一般风管,软风管和风管占位符:从形状上也分为三类 ...

随机推荐

  1. oracle可传输表空间测试

    使用RMAN在恢复表空间的时候,表空间数据文件DBID和恢复数据库的数据文件DBID必须相同 可传输表空间不需要这样,也就是可以快速的把这个表空间插入另一个数据库使用 可传输表空间内的对象必须不依赖与 ...

  2. 如何用Python中自带的Pandas和NumPy库进行数据清洗

    一.概况 1.数据清洗到底是在清洗些什么? 通常来说,你所获取到的原始数据不能直接用来分析,因为它们会有各种各样的问题,如包含无效信息,列名不规范.格式不一致,存在重复值,缺失值,异常值等..... ...

  3. python-列表包字典-根据字典的某一个键的值来进行排序

    python-列表包字典-根据字典的某一个键的值来进行排序 列表包字典的数据结构 要实现按照字典中的某一个键所对应的值进行排序 有两种办法 方法一,使用列表的sort方法 由小到大排 列表.sort( ...

  4. Matlab GUI学习总结

    从简单的例子说起吧.   创建Matlab GUI界面通常有两种方式:   1,使用 .m 文件直接动态添加控件     2.  使用 GUIDE 快速的生成GUI界面显然第二种可视化编辑方法算更适合 ...

  5. 【分享】每个 Web 开发者在 2021 年必须拥有 15 个 VSCode 扩展

    为什么VSCode如此受欢迎 Visual Studio Code在开发人员中迅速流行起来,它是最流行的开发环境,可定制性是其流行的原因之一. 因此,如果你正在使用VSCode,这里有一个扩展列表,你 ...

  6. Redisson 分布式锁实战与 watch dog 机制解读

    Redisson 分布式锁实战与 watch dog 机制解读 目录 Redisson 分布式锁实战与 watch dog 机制解读 背景 普通的 Redis 分布式锁的缺陷 Redisson 提供的 ...

  7. 页面切换提速30%!京东商城APP首屏耗时监控及优化实践

    https://mp.weixin.qq.com/s/vIG_x1MWC33HKV1GxalVHg 原创 平台研发朱跃棕等 京东零售技术 2020-12-09 网络接口请求可以从接口结构合理性及多接口 ...

  8. Python Data Structure and Algorithms Tutorial

    Python - Algorithm Design - Tutorialspoint https://www.tutorialspoint.com/python_data_structure/pyth ...

  9. 唯一ID生成算法剖析

    https://mp.weixin.qq.com/s/E3PGP6FDBFUcghYfpe6vsg 唯一ID生成算法剖析 原创 cloudoxou 腾讯技术工程 2019-10-08    

  10. CF1209A

    所谓染色,并使同颜色数都能被当前颜色中最小的数整除 也就是说,把能被某个数整除的所有数放在一起为一组,问共有几组 开始我想写个并查集但是很懒,看数据范围小的可怜,那我们写个暴力看看 因为每组的共因数都 ...