王健潔 ,趙麗娟,王曉帆,王光昇
(1.天津市測繪院,天津 300381; 2.天津市國土資源測繪和房屋測量中心,天津 300051)
在國內(nèi)的測繪行業(yè)中,AutoCAD 一直是數(shù)據(jù)采集加工的主流軟件,這主要是因為它方便易用,圖形表現(xiàn)靈活多樣,尤其是對線劃圖的表達(dá)更是其他制圖軟件所無法比擬的,所以在數(shù)字化測繪的歷史進(jìn)程中,AutoCAD 扮演了相當(dāng)重要的角色。但在信息化測繪的今天,人們對地圖的要求已經(jīng)不是僅僅局限于圖面了,而對信息的承載、獲取、共享等方面提出了越來越多的需求。在這種情況下,矛盾就出現(xiàn)了,GIS 數(shù)據(jù)長于信息的表達(dá),有很多人便提出了摒棄CAD,直接采集GIS 數(shù)據(jù)的建議,但顯然,直接采集GIS 數(shù)據(jù)的技術(shù)手段并不成熟,同時我們也無法一下子就放棄AutoCAD 的巨大優(yōu)勢。發(fā)展不應(yīng)該只有放棄,需要將CAD 與GIS 進(jìn)行有效地融合,各取所長,尋找更合理的方案使得數(shù)據(jù)的采集滿足信息化的測繪的需求。
天津市測繪院在地形圖數(shù)據(jù)管理中大膽變革,將原來的圖幅管理模式改為單元管理,避免了圖幅接邊問題,保證了數(shù)據(jù)的完整性,方便了外業(yè)修測,但在數(shù)據(jù)分發(fā)時仍需要按圖幅裁切提供,在AutoCAD 中,對點(diǎn)、線的裁切分幅是可以用程序來完成的,但對于Hatch 填充面AutoCAD 并沒有提供區(qū)域裁剪的工具,那么如何實(shí)現(xiàn)GIS 數(shù)據(jù)中的面那樣能夠任意裁剪呢,本文以自定義面域為例,探討了如何將GIS 數(shù)據(jù)的功能有效地融合到AutoCAD 的圖形元素中。
在設(shè)計的面域元素時應(yīng)該考慮到以下幾方面問題。
通過Clipper 類庫實(shí)現(xiàn)面域的空間操作。Clipper類庫是開源的多邊形裁剪庫,它可以實(shí)現(xiàn)多邊形的相交(Intersection)、合并(union)、差異(difference)、異或(xor)、偏移(Offset)等操作,它能夠處理復(fù)雜的多邊形,如自相交、環(huán)島多邊形等,如圖1所示。
圖1 Clipper 類庫
在一般的GIS 數(shù)據(jù)中是不支持圓弧的,而既然設(shè)計在AutoCAD 中表現(xiàn)面域,就需要解決面域中圓弧的裁切問題。
在自定義實(shí)體類時,一般是從AutoCAD 基類派生,所以要選擇適合表現(xiàn)復(fù)雜面的基類,這里選擇從AcDbMPolygon 基類派生。
采用自定義實(shí)體的優(yōu)勢在于,可以對圖形的表現(xiàn)以及與用戶交互的接口進(jìn)行全面的控制,使之符合地形圖要素的要求。例如:
(1)定義編碼屬性
以前通過AutoCAD 對象的thickness 屬性來記錄編碼信息,實(shí)際上這種方法是不合理的,因為thickness 在三維空間視圖中用于表示目標(biāo)的厚度。僅僅是由于thickness 允許用戶修改,同時受限于以往的技術(shù)手段,才勉強(qiáng)這樣去做了,當(dāng)我們用自定義實(shí)體技術(shù)來定制要素的時候,可以解決這個問題。自定義實(shí)體中的屬性和方法可以自由地定制,對編碼屬性而言,用一個字符串類型的成員就可以了,這樣一個簡單設(shè)計可以避免因為使用double 類型值帶來的精度損失。
(2)定義要素名稱
在自定義實(shí)體中,可以在屬性面板中顯示要素的名稱,使得在AutoCAD 中表現(xiàn)的地形圖要素更加直觀,便于圖面判讀,如圖2所示。
圖2 顯示要素名稱
一般創(chuàng)建自定義實(shí)體需要建立兩個工程,一個是DBX 工程,負(fù)責(zé)自定義實(shí)體的讀、寫、顯示等操作;另一個是ARX 工程,包含了程序的入口點(diǎn),用戶在該工程中建立命令函數(shù)創(chuàng)建自定義實(shí)體對象實(shí)例,然后在AutoCAD 中加載運(yùn)行。
為了方便起見,一般要在“項目依賴項”中設(shè)置ARX 工程依賴于DBX 工程,這樣就不用在ARX 工程屬性中明確寫明所依賴的庫文件了。
如果想在屬性面板中顯示自定義的屬性,還必須建立一個COM 工程,為每個自定義實(shí)體類建立COM Wrapper 對象,然后在自定義實(shí)體類中實(shí)現(xiàn)subGetClassID 函數(shù),最后在新生成的* _i.c 頭文件中復(fù)制CLSID 并將其賦值給subGetClassID 函數(shù)的參數(shù)* pClsid,這樣就建立起自定義實(shí)體類和COM 包裝類的關(guān)系,如:
Acad::ErrorStatus CMyPolygon::subGetClassID (CLSID *pClsid)const{
assertReadEnabled ();
* pClsid=CLSID_MyPolygonCom;
return(Acad::eOk);
}
有了COM Wrapper 對象,我們就可以在AutoCAD中通過VisualLISP 創(chuàng)建和編輯自定義對象了。
自定義實(shí)體工程不是必須要建立成DBX 類型,也可以建立成ARX 類型,二者的區(qū)別在于:DBX 工程是標(biāo)準(zhǔn)的自定義實(shí)體工程,可以加載到所有RealDWG 宿主應(yīng)用程序中,如AutoCAD、VoloView 等;ARX 類型的應(yīng)用程序只能加載AutoCAD 中,所以當(dāng)我們的應(yīng)用程序僅僅是應(yīng)用在AutoCAD 環(huán)境中時,完全可以把自定義實(shí)體工程建立成ARX 工程,這樣做的原因是,ARX應(yīng)用程序有很多函數(shù)在DBX 工程中不支持,如acedAlert、acedRedraw 等以“aced”開頭的函數(shù),改成ARX 工程后,我們可以應(yīng)用的功能函數(shù)更多。
在互聯(lián)網(wǎng)上可以很容易下載到Clipper 源程序,將其中的clipper.cpp、clipper.hpp 兩個文件添加到自定義面域的工程中,默認(rèn)情況下Clipper 代碼中使用std::max、std::min 函數(shù)的地方會與C+ +編譯器的全局函數(shù)沖突,導(dǎo)致編譯錯誤,需要在工程屬性的預(yù)處理器定義中添加預(yù)定義編譯開關(guān)NOMINMAX。
假設(shè)定義如下的測試函數(shù),用于實(shí)現(xiàn)兩個CMy-Polygon 對象的裁判剪操作:
Bool DoExecute(CMyPolygon* poly_a,CMyPolygon* poly_b,ClipType clipType,double scale);函數(shù)實(shí)現(xiàn)思路如下:
定義3 個Polygons 變量:
ClipperLib::Polygons sub;
ClipperLib::Polygons clp;
ClipperLib::Polygons sol;
sub 是目標(biāo)對象,clp 是剪切器對象,首先,從poly_a、poly_b 中讀取閉合環(huán)坐標(biāo)填充到sub 和clp 中,需要注意的是,從poly_a 和poly_b 中讀取每個環(huán)的坐標(biāo)時,首尾點(diǎn)是相同的,所以需要去掉最后一個點(diǎn),這樣符合Clipper 類中Polygon 的定義規(guī)則,然后,將兩個多邊形通過AddPolygons 方法填加Clipper 變量中,最后執(zhí)行剪切操作:
Clipper clpr;
clpr.AddPolygons(sub,ClipperLib::ptSubject;
clpr.AddPolygons(clp,ClipperLib::ptClip);
bool succeeded = clpr.Execute(clipType,sol,ClipperLib::pftEvenOdd,ClipperLib::pftEvenOdd);
運(yùn)行的結(jié)果保存在ClipperLib::Polygons 類型的參數(shù)sol 中。
在Clipper::Polygons 中多邊形面中每個閉合路徑是通過點(diǎn)表來描述的,所以對CAD 中的弧段需要進(jìn)行圓弧加密,然后由加密后的點(diǎn)集組成多邊形再進(jìn)行裁剪操作,但是裁剪后的多邊形無法復(fù)原被分隔后的圓弧信息,如圖3所示。
面域1 中的圓弧AB 加密后為綠色線所示,以加密后的點(diǎn)形成的多邊形與面域2 進(jìn)行裁剪,點(diǎn)D 為其中的分割點(diǎn),由點(diǎn)A 到點(diǎn)D 之間的綠色線是裁剪后的一條邊,我們無法從這條邊返回對應(yīng)的圓弧信息,這樣就會造成精度的損失,關(guān)于這種圓弧裁切的問題我們的解決方法是:
圖3 圓弧加密前后的裁剪
首先,通過CAD 中的實(shí)體的相交關(guān)系找出多邊形中的所有弧段,然后計算出每條弧段與另一多邊形的各個邊的交點(diǎn),那么這些交點(diǎn)將弧段分隔成子弧段,然后再將子弧段加密,再通過Clipper 進(jìn)行多邊形運(yùn)算,最后將運(yùn)行結(jié)果中的點(diǎn)按照子弧段進(jìn)行轉(zhuǎn)換,即加密點(diǎn)轉(zhuǎn)換成圓弧。
對圓弧進(jìn)行加密可以直接通過ObjectARX 中的getSamplePoints 函數(shù):
參數(shù)pointArray 存儲從弧起點(diǎn)至終點(diǎn)的所有采集點(diǎn),包括弧起點(diǎn)和終點(diǎn)。參數(shù)approxEps 為弦公差,即偏離一個光滑圓的可接受偏差,它既可用弦角度建立,又可用偏差來建立。弦公差會影響用來在屏幕上畫圓圈的弦線的數(shù)目,弦公差越小,需要繪制的弦數(shù)越多,如圖4所示。
圖4 弦公差
由于已經(jīng)存儲了子弧段,在裁剪后的加密點(diǎn)返回圓弧的過程中只需要將那些落在子弧段上的點(diǎn)去掉可以了。
本文簡要論述了在AutoCAD 中實(shí)現(xiàn)自定義面域的基本方法,并將GIS 中對面域的操作引入到自定義實(shí)體中。在今后的數(shù)據(jù)采集中我們可以繼續(xù)發(fā)揮CAD 在線劃數(shù)據(jù)采集上的優(yōu)勢,同時可以有效地借鑒GIS 在信息表達(dá)上的長處,將兩者融為一體,更好地為信息化測繪服務(wù)。
[1]Autodesk.ObjectARX 開發(fā)指南[R].1999.
[2]老大中,趙占強(qiáng).AutoCAD 2000ARX 二次開發(fā)實(shí)例精粹[M].北京:國防工業(yè)出版社,2001.
[3]Charles McAuley.AutoCAD 2000ObjectARX 編 程 指 南[M].北京:機(jī)械工業(yè)出版社,2000.
[4]張長勛.AutoCAD VisualLISP 程序開發(fā)技術(shù)[M].北京:國防工業(yè)出版社,2005.
[5]余承飛,方勇.AutoCAD 2000 二次開發(fā)技術(shù)(ObjectARX)[M].北京:人民郵電出版社,1999.
[6]宋延杭,王川,李永宣.ObjectARX 實(shí)用指南[M].北京:人民郵電出版社,1999.