应用Automation技术进行AutoCad的开发
Autodesk公司的AutoCad软件广泛的应用于建筑、机械等设计领域。众所周知,AutoCad是一种极其灵活的应用系统,用户可以通过编程的方式对其进行定制。在以往的AutoCad系统开发中,最常用的是AutoLisp和ADS,但AutoLisp不如编程语言方便,在开发较大项目时力不从心,而ADS虽由功能强大的C语言编制,但较为复杂,不适应当前可视化编程的需要。幸运的是,在最新推出的AutoCad r14版本中,Autodesk公司对AutoCad本身加入了ActiveX自动化服务功能(ActiveX Automation server capabilities),使得用户可以通过可视化编程工具,如Visual Basic、Delphi等对AutoCad进行系统开发,极大的提高了工作效率。在最近为建筑公司开发的一套建筑施工三维演示系统中,我们采用AutoCad为平台,Visual Basic为编程工具,成功的应用ActiveX对AutoCad进行了二次开发。本文将从编程实践的角度对使用VB对AutoCad控制的技术及遇到的问题进行简要的说明。
一、 AutoCad的对象模型(AutoCad Object Model)
如果一个应用程序支持自动化,那么其他应用程序就可以通过其暴露的对象(Object)对其自动操作。在本例中,我们开发的程序为客户机,AutoCad是服务器,应用程序正是通过对AutoCad暴露的各级对象进行操作而控制AutoCad工作的。对象本身包含自己的方法和属性。通过方法可以实现对象的一些操作,而对象状态信息的收集或改变则是通过属性完成。例如,直线是AutoCad中的一个对象,它的属性可以是线形、起始点坐标、所在层等性质,方法则有拷贝、镜像等操作。
理解AutoCad的对象模型是对其编程的基础。AutoCad以层次结构组织对象。在顶层是Application对象(即AutoCad 本身),其他对象均为Application对象的子对象。在Application对象下面是Preferences(优先设置)和Document(文档)对象,通过Preferences对象可以对AutoCad Tools>Preferences菜单项中的几乎每一个选项进行访问和修改,以获取或改变AutoCad的优先设置。Document对象是控制AutoCad图形文件的直接对象,它代表某一个装入的CAD图形文件(一般设为当前激活的文件)。Document对象下面有Model Space(模型空间)和Paper Space(图纸空间)对象及Blocks(块)、Layers (层)、Plot(出图)、Selectionsets(选择集)、 views(视图)、 utility(功能)等一系列对象(集合),其含义与AutoCad中相似。Model Space是当前图形文件中图形实体,如直线、圆、多义线等的集合,每个实体即是一个对象,可通过属性和方法改变实体或生成新实体。
对非图形实体,如层(layer)、线形(line type)等的访问则通过访问Document对象下面的相应的集合类型的子对象,如Layers 、LineTypes等来实现。集合类型的对象可以使用VB中所有的集合操作方法。Plot对象提供了访问Plot对话框中各选项的桥梁,使应用程序具有用不同方式控制AutoCad出图的能力。Utility对象使用户在AutoCad命令行与CAD交互成为可能,通过它可以处理整型、浮点型、字符型等用户输入,还可以接受点(Point)或角(Angle)等AutoCad的特殊量。
二、 AutoCad对象的使用
在本节中,将结合我做项目得到的一些经验以例程的方式对AutoCad的常用对象及其使用进行讨论。
1.开始一个应用程序
如前所述,Application对象位于AutoCad层次对象结构的顶层,它代表AutoCad本身,用户的应用程序也理所当然从Appliction对象的建立开始。
Dim acadapp As Object 建立Application对象
Dim acaddoc As Object 建立Document对象
Dim mospace As Object 建立Model Space 对象
On Error Resume Next
Set acadapp = GetObject(, "autocad.application") ‘若AutoCad
已启动,则直接得到
If Err Then
Err.Clear
Set acadapp = CreateObject("autocad.application") ‘若
AutoCad未启动,则运行它
If Err Then
MsgBox Err.Description
Exit Sub
End If
End If
acadapp.Visible = True ‘使AutoCad可见
Set acaddoc = acadapp.ActiveDocument ‘设acaddoc为当前
图形文件
Set mospace = acaddoc.ModelSpace ‘设mospace为当前图形
文件的模型空间
以上程序段是应用程序初始化的过程,一般对AutoCad图形文件的操作,主要是与Application、Document和Model Space等对象发生关系。
Application对象是一系列对象的父对象,可以通过它的属性设置来改变AutoCad的窗口设置。请看下面代码:
acadapp.Top=100 设置AutoCad窗口的位置
acadapp.Left=200
acadapp.Height=1000 调整AutoCad窗口的大小
acadapp.Width=800
acadapp.Caption="my first application" 设置AutoCad窗口的
标题
通过Application对象的方法还可以方便的调入ADS或ARX程序,以利于各类程序的集成。其例程为:
Dim arxname As string
acadapp.LoadARX arxname ‘arxname即为调入的arx程序名(带 路径)
2.通过Document对象对图形文件的操作
Document对象提供了大多数AutoCad的文件功能,可以通过它实现对文件的更新(New)、打开(Open)、输出(Export)、输入(Import)等操作,一般要先把Document对象设为Application对象的 ActiveDocument属性,以返回当前图形文件。
Set acaddoc=Application.ActiveDocument
请看下面的例子对文件的操作:
Dim dwgname As String
dwgname = "c:\acadr14\sample\campus.dwg"
If Dir(dwgname) <> "" Then
acaddoc.Open dwgname 打开一个CAD文件
Else
acaddoc.new("acad") 以acad.dwt为模板建立一个新
文件
End If
Document对象还提供了两个十分有用的方法——SetVariable 和 GetVariable,通过它们可以得到或改变AutoCad的系统变量。
如:acaddoc.SetVariable "Orthomode", 1 打开正交模式
dim cadver As String
cadver=acaddoc.Getvariable("Acadver") 获取AutoCad的版本号
4.与用户交互
Utility对象提供了与用户在命令行交互的途径,可以让用户输入数字、字符串及角度、点坐标等参量。下面说明如何应用Utility交互替代AutoCad命令中的提示:
Dim acadUtil as Object
Dim stPnt, enPnt As Variant
Dim prompt1, prompt2 As String
Set acadUtil=acaddoc.Utility 设置Utility对象
prompt1 = "起始点: " ‘代替from Point
prompt2 = "终止点: " 代替End Point
stPnt = acadUtil.GetPoint(, prompt1)
enPnt = acadUtil.GetPoint(stPnt, prompt2)
获得用户输入(既可输入坐标值,也可直接在屏幕上选点)
Dim startPoint(0 To 2) As Double
Dim endPoint(0 To 2) As Double
startPoint(0) = stPnt(0)
startPoint(1) = stPnt(1)
startPoint(2) = stPnt(2)
endPoint(0) = enPnt(0)
endPoint(1) = enPnt(1)
endPoint(2) = enPnt(2)
moSpace.AddLine startPoint, endPoint 利用用户输入生成直线
把系统变量设置SetVariable与Utility对象的GetString方法结合,即可向AutoCad的状态行写入内容:
Dim yourname as String
yourname = acadUtil.GetString(0, " 请输入您的姓名: ")
acaddoc.SetVariable "MODEMACRO", yourname & ", 你好!"
5.对非图形对象的操作
非图形对象如层(Layers)、视图(Viewports)、坐标系(UCSs)、块 (Blocks)等与图形实体集合ModelSpace、PaperSpace同是Document对象的子对象,它们本身既是对象,又是对象的集合,如Layers是当前打开的图中所有层的集合,使用Add方法来建立新层,并可以遍历所有层,通过改变其属性达到关闭(Off)、冻结层(Freeze)的目的.
①把层名为"wall"的层冻结,打开层名为"beam"的层,并设为当前层
Dim tlayer as Object
For Each tlayer In acaddoc.Layers
If tlayer.Name = "wall" Then
tlayer.Freeze = acTrue
Else If tlayer.Name="beam" Then
tlayer.LayerOn = acTrue
Set acaddoc.ActiveLayer = tlayer
End If
Next
②创建名为"myview"的新视图
可以通过ActiveX自动实现变换视图角度及缩放全图。
Public Sub changeview(ByVal x, ByVal y, ByVal z)
Dim newDirection(0 To 2) As Double
Dim vport As Object
acaddoc.ActiveSpace = acModelSpace ‘使ModelSpace成为活动
空间
Set vport = acaddoc.Viewports.Add("newview") ‘建立新视图
newDirection(0) = x
newDirection(1) = y
newDirection(2) = z ‘视图的视角方向
vport.Direction = newDirection
acaddoc.ActiveViewport = vport ‘把新视图激活
acaddoc.ActiveViewport.ZoomAll ‘全图显示
End Sub
以上例程是对Layers、Viewports对象的举例,其他非图形对象的引用与此类似。
6.对选择集的操作
在对AutoCad的编程中,选择集占有十分重要的地位,对编程者而言,并不清楚图中包含什么实体,只有通过用户的选择或通过过滤条件把所需的实体加入选择集,再对选择集中的实体进行操作。下面例程给出了两种筛选建立选择集的方法,把图中所有在层"wall"上的直线亮显。
①由用户在屏幕上选择实体
Dim tempset as Object
Dim obj as Object
Set tempset = acaddoc.SelectionSets.Add("newset") 建立新选择集
tempset.SelectOnScreen ‘用户在屏幕上选择
For Each obj In tempset ‘遍历选择集中的实体
If obj.EntityName="AcDbLine" And obj.Layer="wall" Then
obj.HighLight(True) 亮显实体
End IF
Next
这种选择方式给用户较大的自由,但不能保证选择集内包含所有我们期望的实体,若要精确过滤出所需实体,应该给选择集加入条件。
②使用过滤器(Filter)筛选实体
Dim actualCode(3) As String
Dim actualValue(3) As String
Dim groupcode As Variant
Dim groupValue As Variant
Dim extminpt(2) As Double
Dim extmaxpt(2) As Double
Dim tsset As Object
Dim tobj As Object
actualCode(0) = -4
actualValue(0) = " actualCode(1) = 8
保证 Layer是"wall"
actualValue(1) = "wall"
actualCode(2) = 100
actualValue(2) = "AcDbLine" 所选实体为直线
actualCode(3) = -4
actualValue(3) = "AND>"
.SendKeys的妙用
AutoCad的ActiveX虽然强大,但不是所有问题都可以通过它解决。要在VB中使用AutoCad对象没有的方法,就须用到VB中的过程SendKeys。通过SendKeys把AutoCad的命令行如同批处理一样送到AutoCad中自动执行,在效果上与使用对象的方法是相同的。另外,还可以使用简单的AutoLisp语言增强AutoCad命令行的功能。下例是执行break命令而编写的过程。其中的(handent"***")是从Lisp语言中借来的,可以直接在命令行通过实体句柄(Handle)来确定实体。
SendKeys "{esc}", True
SendKeys "{esc}", True ‘避免以前命令的干扰
SendKeys "_break" & "{enter}", True
SendKeys "{(}" & "handent" & """" & wallhandle & """" & "{)}" & "{enter}", True ‘选择要断开的实体(wallhandle为其句柄)
SendKeys Format(cood1(0)) & "," & Format(cood1(1)) & "{enter}", True
SendKeys Format(cood2(0)) & "," & Format(cood2(1)) & "{enter}", True ‘cood1与cood2是实体上断开点的坐标
三、 最终的补充说明
1. 尽量采用迭代的方法遍历集合
如前所述,对CAD的编程中涉及大量的集合操作,下面的代码段与迭代法效果相同,但效率较低。
Dim I As Integer
For I=0 To sset.Count-1
sset(I).HighLight(True)
Next I
在创建自己的集合时,关键字尽量采用Handle值,以便查找,并可通过HandletoObject方法将Handle值转化为实体(Object)
2. 采用AutoCad r14.01版
Autodesk公司在r14版中加入了ActiveX Automation,但尚不完善。在随后推出的14.01版中,Autodesk公司解决了r14版的不少错误,使得应用程序运行更为流畅、稳定。如果要进行AutoCad的ActiveX编程,建议采用VB5.0和AutoCad r14.01。
3. 关于ActiveX的资源
国内有不少介绍ActiveX编程及AutoCad的资料,但迄今还未发现有完整讨论AutoCad的ActiveX编程的书籍,目前最容易得到的资料就是Autodesk公司编写的随机帮助,其中有一章“ActiveX Automation",有全部对象的方法、属性说明。 |