OpenCASCADE DataExchange DWG

eryar@163.com

Abstract. DWG is a file format created in the 70’s for the emerging CAD applications. Currently it is the native file format of AutoCAD, a proprietary CAD program developed by Autodesk. Libredwg is a free C library to read and write DWG files. This program is part of GNU project, released under the aegis of GNU. The paper focus on the usage of Libredwg, and use the Libredwg to read a DWG file and output the entities of the DWG to Tcl script for OpenCASCADE Draw Test Harness visualization.

Key Words. OpenCASCADE, DWG, Libredwg, DataExchange, Windows

1. Introduction

DWG是CAD软件AutoCAD及基于AutoCAD的软件保存设计数据所用的一种专有文件格式,始于1970年代的一套Interact CAD软件。之后Autodesk公司于1982年取得版权开始使用这种文件格式。Autodesk公司拥有、开发并且更新DWG文件格式,通常每隔几年DWG就会随着在AutoCAD中添加新的特性而对DWG格式进行更新。

DWG格式及它的ASCII格式变体DXF格式,已经成为CAD制图数据交换中的事实文件标准,据估计全世界有超过十亿个DWG文件。有几家公司正在对DWG文件格式进行逆向工程以试图为其它的设计软件提供读写DWG文件的能力。Autodesk公司也提供了一套需要授权的DWG读写技术开发包“RealDWG”。

新版的AutoCAD可以打开旧版的DWG文件,AutoCAD2007可以打开2.0版本的DWG文件并且可以保存为R14版本。另外Autodesk公司提供一个免费的DWG查看工个“DWG TrueView”用于查看所有版本的DWG文件。另外Autodesk公司是vendor lock-in策略的强力支持者,尽力保护DWG文件格式并且禁止开发支持DWG格式的开放源代码库。

2006年11月12日,Autodesk公司对Open Design Alliance-一款支持DWG格式的自由库OpenDWG提出了诉讼。

ASCII格式的DXF文件的文档Autodesk提供了,但是二进制的DWG格式并没有提供相关文档,由上可见对DWG文件的读写处理是非常困难的。本文主要介绍如何使用Libredwg库来对读取DWG中的几何数据,并将几何数据生成为可以在OpenCASCADE中显示的Tcl脚本,以验证读取数据的正确性。

2.Modify Libredwg for Visual Studio

Libredwg是一个Free的读写DWG文件的C库,这个程序是GNU项目的一部分,授权方式是GNU GPL3。

Libredwg是Libdwg的一个分支,其目的是创建OpenDWG库的一个替代库。也是高优先级的Free软件项目:

http://www.fsf.org/campaigns/priority-projects/priority-projects/highpriorityprojects#ReplaceOpenDWG ,更多信息可访问http://www.gnu.org/software/libredwg 。

从网上下载的Libredwg源程序是在Linux下编译的,并没有配置在Windows下编译方法。为了使用Libredwg可以在Windows上的Visual Studio中编译通过,对Libredwg做了一些修改,最终编译成功。在Visual Studio 2008上成功编译的工程可以文后的链接中下载。

下面给出使用Libredwg读取DWG文件中直线、圆及文字的例子程序:

/*
* load_dwg.c: load a DWG, get lines, text and circles
* written by Felipe Castro
* modified by Felipe Corrêa da Silva Sances
* modified by Thien-Thi Nguyen
*/ #include <dwg.h>
#include "suffix.c" void add_line(double x1, double y1, double x2, double y2)
{
// Make something with that
} void add_circle(double x, double y, double R)
{
// Make something with that
} void add_text(double x, double y, char *txt)
{
// Make something with that
} int load_dwg(char *filename)
{
unsigned int i;
int success;
Dwg_Data dwg; dwg.num_objects = ;
success = dwg_read_file(filename, &dwg);
for (i = ; i < dwg.num_objects; i++)
{
Dwg_Entity_LINE *line;
Dwg_Entity_CIRCLE *circle;
Dwg_Entity_TEXT *text; switch (dwg.object[i].type)
{
case DWG_TYPE_LINE:
line = dwg.object[i].tio.entity->tio.LINE;
add_line(line->start.x, line->end.x, line->start.y, line->end.y);
break;
case DWG_TYPE_CIRCLE:
circle = dwg.object[i].tio.entity->tio.CIRCLE;
add_circle(circle->center.x, circle->center.y, circle->radius);
break;
case DWG_TYPE_TEXT:
text = dwg.object[i].tio.entity->tio.TEXT;
add_text(text->insertion_pt.x, text->insertion_pt.y, text->text_value);
break;
}
}
dwg_free(&dwg);
return success;
} int main (int argc, char *argv[])
{
REQUIRE_INPUT_FILE_ARG (argc);
load_dwg (argv[]);
return ;
}

因为Libredwg用的C编程风格,没有定义导出定义宏,所以决定将Libredwg编译成静态库libredwg.lib,然后使用其头文件及这个静态库的方式来在程序中使用Libredwg库。

经过测试,若DWG中只有简单的线和圆等简单实体,Libredwg还是可以正确读取出。但是若用rewrite的例子来测试写DWG的功能,简单的实例如一个圆的数据都会写出到DWG失败,看样子写的功能还没有完全实现好。

3.DWG to OCC

基于上面的例子程序,结合Libredwg的读取功能,将DWG中的几何数据导出成Tcl脚本,这样就可以方便在OpenCASCADE的Draw Test Harness中来测试结果了。下面给出具体的程序实例及如何在Draw Test Harness中来使用生成的Tcl脚本。

/*
* Copyright (c) 2014 eryar All Rights Reserved.
*
* File : Main.cpp
* Author : eryar@163.com
* Date : 2014-10-15 20:46
* Version : 1.0v
*
* Description : Use libredwg to read data from DWG and
* output them to Tcl script for Draw.
*
* Key words : OpenCASCADE, libredwg, Draw Test Harness
*/ #include "dwg.h" #include <fstream>
#include <iostream> #pragma comment(lib, "../Debug/libredwg.lib") // Output the entities to Tcl for OpenCASCADE Draw.
static std::ofstream theTclExporter("d:/dwg2occ.tcl"); void OutputLine(int id, const Dwg_Entity_LINE* theLine)
{
// Draw Tcl command: vline name xa ya za xb yb zb
theTclExporter << "vline line" << id << " "
<< theLine->start.x << " " << theLine->start.y << " " << theLine->start.z << " "
<< theLine->end.x << " " << theLine->end.y << " " << theLine->end.z << std::endl;
} void OutputCircle(int id, const Dwg_Entity_CIRCLE* theCircle)
{
// Draw Tcl command: circle name x y [z [dx dy dz]] [ux uy [uz]] radius
// 1. make a curve
theTclExporter << "circle circle" << id << " "
<< theCircle->center.x << " " << theCircle->center.y << " " << theCircle->center.z << " "
<< "0 0 1 " << theCircle->radius << std::endl; // 2. make edge from the circle
theTclExporter << "mkedge e" << id << " "
<< "circle" << id << std::endl; // 3. display the circle edge
theTclExporter << "vdisplay e" << id << std::endl;
} void OutputText(int id, const Dwg_Entity_TEXT* theText)
{
// vdrawtext : vdrawtext : vdrawtext name X Y Z R G B hor_align ver_align angle zoomable height Aspect [Font [isMultiByte]]
theTclExporter << "vdrawtext " << theText->text_value << " "
<< theText->insertion_pt.x << " " << theText->insertion_pt.y << ""
<< " 255 255 000 "
<< theText->horiz_alignment << " " << theText->vert_alignment << " "
<< theText->height << " 1 Times-Roman"<< std::endl;
} int LoadDwg(char* theDwgFile)
{
int aResult = ; Dwg_Data aDwgData; aResult = dwg_read_file(theDwgFile, &aDwgData); for (unsigned int i = ; i < aDwgData.num_objects; ++i)
{
switch (aDwgData.object[i].type)
{
case DWG_TYPE_LINE:
OutputLine(i, aDwgData.object[i].tio.entity->tio.LINE);
break; case DWG_TYPE_CIRCLE:
OutputCircle(i, aDwgData.object[i].tio.entity->tio.CIRCLE);
break; case DWG_TYPE_TEXT:
OutputText(i, aDwgData.object[i].tio.entity->tio.TEXT);
break;
}
} return aResult;
} int main(int argc, char* argv[])
{
theTclExporter.flags(std::ios::fixed); theTclExporter << "pload ALL" << std::endl;
theTclExporter << "vinit" << std::endl;
theTclExporter << "vtrihedron tr" << std::endl; if (argc < )
{
std::cout << "please input the dwg file name!" << std::endl;
}
else
{
LoadDwg(argv[]);
} theTclExporter << "vdisplayall" << std::endl;
theTclExporter << "vfit" << std::endl;
theTclExporter << "vhelp" << std::endl; return ;
}

将生成的dwg2occ.tcl在OpenCASCADE的Draw Test Harness中显示如下所示:

Figure 3.1 Import the Tcl script in the Draw Test Hanress of OpenCASCADE

Figure 3.2 Enitites in the DWG file

Figure 3.3 Entities in the Draw Test Harness of OpenCASCADE

通过对比发现,直线和圆已经正确读出,但是文字没有读出来。看来Libredwg的可靠性还有待提高啊。

4.Conclusion

通过使用Libredwg来读取DWG格式中几何数据,并将其转换成OpenCASCADE的Draw Test Harness中能执行的Tcl脚本,以方便测试libredwg读取数据的正确性。

通过简单测试发现,libredwg能读取直线和圆,但是文字内容没有正确读出,看来libredwg的可靠性还有待提高,但是发现这个开源库的更新很缓慢。

5. References

1. DWG Wiki: http://en.wikipedia.org/wiki/.dwg

2. Libredwg: http://www.gnu.org/software/libredwg

3. OpenCASCADE Test Harness User’s Guide 6.7.1

Libredwg for Visual Studio: OpenCASCADE DataExchange DWG

OpenCASCADE DataExchange DWG的更多相关文章

  1. Open Cascade DataExchange DXF

    Open Cascade DataExchange DXF eryar@163.com 摘要Abstract:对DXF文本格式进行详细介绍,并介绍了如何使用开源库dxflib对DXF文件进行读写操作, ...

  2. OpenNURBS to OpenCASCADE

    OpenNURBS to OpenCASCADE eryar@163.com Abstract. The OpenNURBS initiative provides CAD/CAM/CAE and c ...

  3. Qt with OpenCascade

    Qt with OpenCascade 摘要Abstract:详细介绍了如何在Qt中使用OpenCascade. 关键字Key Words:Qt.OpenCascade 一.引言 Introducti ...

  4. Building OpenCascade on Windows with Visual Studio

    Building OpenCascade on Windows with Visual Studio eryar@163.com 摘要Abstract:详细说明OpenCascade的编译配置过程,希 ...

  5. The Installation and Compilation of OpenCASCADE

    OpenCASCADE的编译 The Installation and Compilation of OpenCASCADE eryar@163.com 一. 安装OpenCASCADE 可以从Ope ...

  6. OpenCascade HLR for Pipe Model

    OpenCascade HLR for Pipe Model eryar@163.com 摘要Abstract:在工厂辅助设计(Plant Design)或船舶辅助设计(Ship Design)等CA ...

  7. Open Cascade DataExchange IGES

    Open Cascade DataExchange IGES eryar@163.com 摘要Abstract:本文结合OpenCascade和Initial Graphics Exchange Sp ...

  8. OpenCascade BRep Format Description (2)

    OpenCascade BRep Format Description eryar@163.com 摘要Abstract:本文结合OpenCascade的BRep格式描述文档和源程序,对BRep格式进 ...

  9. OpenCASCADE入门指南

    OpenCASCADE入门指南 eryar@163.com 一.概述 荀子说“君子性非异也,善假于物也”.当你会用英语,就可以与世界各国的人交流:当你会用编程语言,就可以与计算机交流:当你会用数学语言 ...

随机推荐

  1. Ubuntu13.10下安装HADOOP

    2013-03-05 09:04 995人阅读 评论(0) 收藏 举报 运行这个脚本: #/bin/sh sudo add-apt-repository ppa:webupd8team/java su ...

  2. android 5.0以上通知栏、状态栏图标变成白色

    在5.0以上的系统上发现,平常的自定义notification出来的icon,居然在状态栏上变成了纯白色的icon. 看源代码会发现: protected void applyColorsAndBac ...

  3. d8fs9f

    你好 - Helloworld 1. a 2. b 3. c 来自为知笔记(Wiz)

  4. uva12063数位dp

    辣鸡军训毁我青春!!! 因为在军训,导致很长时间都只能看书yy题目,而不能溜到机房鏼题 于是在猫大的帮助下我发现这道习题是数位dp 然后想起之前讲dp的时候一直在补作业所以没怎么写,然后就试了试 果然 ...

  5. [BZOJ4200][Noi2015]小园丁与老司机

    4200: [Noi2015]小园丁与老司机 Time Limit: 20 Sec  Memory Limit: 512 MBSec  Special JudgeSubmit: 106  Solved ...

  6. .net开源后可以查看的源代码

    通过此网站可以直接查询基于.net framework4.6的源码 也可以下载到本地通过VS 查看.

  7. Java 应该跨四个平台

    编程语言从属于操作系统,要统一,就要在根本处统一,要统一的是操作系统,而不是编程语言.你认为是苹果决定苹果树,还是苹果树决定苹果? 编程语言跨操作系统是错误的道路,你见过苹果长在桔子树上的吗?苹果长得 ...

  8. MongoDB常用操作命令大全

    成功启动MongoDB后,再打开一个命令行窗口输入mongo,就可以进行数据库的一些操作.输入help可以看到基本操作命令,只是MongoDB没有创建数据库的命令,但有类似的命令 如:如果你想创建一个 ...

  9. java分享第十六天( java读取properties文件的几种方法&java配置文件持久化:static块的作用)

     java读取properties文件的几种方法一.项目中经常会需要读取配置文件(properties文件),因此读取方法总结如下: 1.通过java.util.Properties读取Propert ...

  10. Python join()函数

    今天写python 100例时,有个题目是大致是这样的:已知输入形式是1+3+2+1,要求输出形式为1+1+2+3 一开始思路是将输入的字符串用split()函数划分成数组,在对数组进行排序,再用fo ...