CATIA Automation 编程初探

最近开始对 CATIA 进行一些自动化的开发,本来想找 CAA 来进行研究,可惜一直没时间和机会去找,暂时就利用 CATIA Automation 来开发了。

由于利用 VB 或 VB.NET 将CATIA 脚本转化为程序确实方便,不过利用C++来操作更适合于我这种开发者。

方法有好多种,这里以画圆作为一个简单例子。

1 利用 Idispatch 接口来编程

        HRESULT hr;
        CLSID AppClsid;
        Idispatch *pApp;

        ::CoInitialize(NULL);
        
	//
	::CLSIDFromProgID (L"CATIA.Application",&AppClsid); //get the unique id of CATIA
        
		
	//
	hr = CoCreateInstance(AppClsid,NULL,CLSCTX_LOCAL_SERVER,IID_Idispatch,(void**)&pApp);


        VARIANT result,buffer;
        Variantinit(&result);


        VARIANT arg2;
        Variantinit(&arg2);
        arg2.vt = VT_BOOL;
        arg2.boolVal = TRUE;
        hr = AutoWrap(disPATCH_PROPERTYPUT|disPATCH_METHOD,&result,pApp,L"Visible",1,arg2);


        hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,L"Documents",0);//here there is no argument,so we put 0
        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *documents = buffer.pdispVal;

        VARIANT arg1;
        Variantinit(&arg1);
        arg1.vt = VT_BSTR;
        arg1.bstrVal = ::SysAllocString(L"Part");
        //VARIANT result,buffer;
        Variantinit(&result);
        hr = AutoWrap(disPATCH_METHOD,documents,L"Add",arg1);

        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *partDocument = buffer.pdispVal;

        //
        hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,partDocument,L"Part",0);
        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *part = buffer.pdispVal;

        //
        hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,part,L"Bodies",0);
        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *bodies = buffer.pdispVal;

        //
        VARIANT arg;
        Variantinit(&arg);
        arg.vt = VT_BSTR;
        arg.bstrVal = ::SysAllocString(L"PartBody");
        hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,bodies,L"Item",arg);
        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *body = buffer.pdispVal;

        //
        hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,body,L"Sketches",0);
        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *sketches1 = buffer.pdispVal;

        //
        hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,L"OriginElements",0);
        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *originElements = buffer.pdispVal;

        //
        hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,originElements,L"PlaneXY",0);
        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *reference1 = buffer.pdispVal;

        //
        VARIANT arg3;
        Variantinit(&arg3);
        arg3.vt = VT_disPATCH;
        arg3.pdispVal = reference1;
        hr = AutoWrap(disPATCH_METHOD,sketches1,arg3);
        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *sketch1 = buffer.pdispVal;

        ////
        //SAFEARRAY * psa;
        //SAFEARRAYBOUND rgsabound[1];

        //rgsabound[0].lLbound = 0;
        //rgsabound[0].cElements = 9;

        //psa = SafeArrayCreate(VT_VARIANT,rgsabound);
        //


        //hr = SafeArrayAccessData(psa);


        //hr = SafeArrayUnaccessData(psa);

        //VARIANT array;
        //Variantinit(&array);
        //array.vt = VT_ARRAY;
        //array.pparray = &psa;
        //
        //hr = AutoWrap(disPATCH_METHOD,sketch1,L"SetAbsoluteAxisData",array);
        VARIANT arg4;
        Variantinit(&arg4);
        arg4.vt = VT_disPATCH;
        arg4.pdispVal = sketch1;
        hr = AutoWrap(disPATCH_PROPERTYPUT|disPATCH_METHOD,L"InWorkObject",arg4);
        //buffer.vt = VT_disPATCH;
        //buffer.pdispVal = result.pdispVal;
        //Idispatch *sketch1 = buffer.pdispVal;

        //
        hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,L"OpenEdition",0);
        buffer.vt = VT_disPATCH;
        buffer.pdispVal = result.pdispVal;
        Idispatch *factory2D1 = buffer.pdispVal;


        //
        VARIANT a1,a2,a3;
        Variantinit(&a1);
        Variantinit(&a2);
        Variantinit(&a3);
        a1.vt = VT_R4;a1.fltVal = 50.f;
        a2.vt = VT_R4;a2.fltVal = 0.f;
        a3.vt = VT_R4;a3.fltVal = 0.f;
        hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,factory2D1,L"CreateClosedCircle",3,a1,a3);
		
        //
	hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,L"CloseEdition",0);
		
	//
	hr = AutoWrap(disPATCH_PROPERTYGET|disPATCH_METHOD,L"Update",0);
		

       if (pApp) pApp->Release();
       ::CoUninitialize();

2 利用CATIA自带的TLB转化成 tlh/tli 来编程

先用VC的 #import 功能进行转化,主要的几个 TLB 为:

InfTypeLib.tlb

KweTypeLib.tlb

PSTypeLib.tlb

MecModTypeLib.tlb

CATGitTypeLib.tlb

试验的 CATIA V5 R18 版本在转换之后有些小错误,之后就可以方便地写代码了。VC6的话,需要把转换后的UTF8编码文件转换成ANSI。

int main(int argc,char* argv[])
{
    ::CoInitialize(NULL);

    ApplicationPtr catia;

    HRESULT hr = S_OK;

    hr = catia.GetActiveObject("CATIA.Application");

    if (Failed(hr))
    {
        hr = catia.CreateInstance("CATIA.Application");
    }

    catia->PutVisible(VARIANT_TRUE);

    //
    DocumentsPtr documents;
    documents = catia->GetDocuments();
    BSTR AddPart = _com_util::ConvertStringToBSTR("Part");
    PartDocumentPtr partDocument;
    partDocument = documents->Add(&AddPart);

    PartPtr part = partDocument->GetPart();
     BodiesPtr bodies = part->GetBodies();
     BodyPtr body = part->GetMainBody();
     HybridBodiesPtr hybridBodies = part->GetHybridBodies();

    FactoryPtr shapeFactory = part->GetShapeFactory();

     HybridShapeFactoryPtr hybridShapeFactory = part->GetHybridShapeFactory();

    SketchesPtr sketches = body->GetSketches();

    OriginElementsPtr originElements = part->GetoriginElements();

    AnyObjectPtr planeXY = originElements->GetPlaneXY();

    ReferencePtr r1 = part->CreateReferenceFromObject(planeXY);

    SketchPtr sketch = sketches->Add(r1);

    part->PutInWorkObject(sketch);

    Factory2DPtr factory2D = sketch->OpenEdition();

    factory2D->CreateClosedCircle(0.0,0.0,50.0);

    sketch->CloseEdition();

    part->Update();

    ::CoUninitialize();

    return 0;
}


3 利用 CAA

还没有找到CAA安装程序,以后再续

相关文章

Format[$] ( expr [ , fmt ] ) format 返回变体型 format$ 强...
VB6或者ASP 格式化时间为 MM/dd/yyyy 格式,竟然没有好的办...
在项目中添加如下代码:新建窗口来显示异常信息。 Namespace...
转了这一篇文章,原来一直想用C#做k3的插件开发,vb没有C#用...
Sub 分列() ‘以空格为分隔符,连续空格只算1个。对所选...
  窗体代码 1 Private Sub Text1_OLEDragDrop(Data As Dat...