ObjectArx 自定义实体

2023-10-29

1.arx文档中规定的必须重写的几个函数

//AcDbObject:

virtual Acad::ErrorStatus 
dwgInFields(AcDbDwgFiler* filer);
 
virtual Acad::ErrorStatus 
dwgOutFields(AcDbDwgFiler* filer) const;
 
virtual Acad::ErrorStatus 
dxfInFields(AcDbDxfFiler* filer);
 
virtual Acad::ErrorStatus 
dxfOutFields(AcDbDxfFiler* filer) const;

//AcDbEntity

virtual Adesk::Boolean      
subWorldDraw(
    AcGiWorldDraw* mode);
 
virtual Acad::ErrorStatus   
subGetGeomExtents(
    AcDbExtents& extents) const;
 
virtual Acad::ErrorStatus   
subTransformBy(
    const AcGeMatrix3d& xform);
 
virtual Acad::ErrorStatus   
subGetTransformedCopy(
    const AcGeMatrix3d& xform,
    AcDbEntity*& ent) const;
 
virtual Acad::ErrorStatus   
subGetGripPoints(
    AcGePoint3dArray& gripPoints,
    AcDbIntArray&  osnapModes,
    AcDbIntArray&  geomIds) const;
 
virtual Acad::ErrorStatus 
subMoveGripPointsAt( 
    const AcDbIntArray& indices,
    const AcGeVector3d& offset);
以上函数是文档中规定的必须重写的函数
2.实际工作中要求的必须重载
1)实际工作中必须重载的函数
virtual Adesk::Boolean      
subWorldDraw(
    AcGiWorldDraw* mode);
分析:从函数的名称我们就可以判断出,这个是自定义实体的外观,相信如果外观都不需要的情况在实际工作中应该是很少出现的
应用情形:
最典型的当属于Jig,只是为了实现动态效果的实体,并不需要自定义实体存档,说白了就是只起到一个临时实体的作用(为什么称作临时实体,只在Jig运行期间出现,Jig完成之后立即释放),在这种情形下,其余文档中要求的重载都可不必实现
3.实例讲解
我们只写一个最简单的实例,来讲解一下自定义实体每一个被重写的函数的意义
自定义实体要实现的效果:
绘制一个圆,并且能够设置颜色
1).h
class CCustomColorCircle:public AcDbEntity
{
public:
   ACRX_DECLARE_MEMBERS(CCustomColorCircle);
   CCustomColorCircle(AcGePoint3d centerPt);
   ~CCustomColorCircle() ;
proctected:
  virtual Acad::ErrorStatus dwgInFields(AcDbDwgFiler* filer);
virtual Acad::ErrorStatus dwgOutFields(AcDbDwgFiler* filer) const;
  virtual Acad::ErrorStatus dxfInFields(AcDbDxfFiler* filer);
 
  virtual Acad::ErrorStatus dxfOutFields(AcDbDxfFiler* filer) const;
  virtual Adesk::Boolean      subWorldDraw(AcGiWorldDraw* mode);

  virtual Acad::ErrorStatus   subGetGeomExtents(AcDbExtents& extents) const;
 
  virtual Acad::ErrorStatus   subTransformBy(const AcGeMatrix3d& xform);
 
  virtual Acad::ErrorStatus   subGetTransformedCopy(const AcGeMatrix3d& xform,AcDbEntity*& ent) const;
 
  virtual Acad::ErrorStatus   subGetGripPoints(AcGePoint3dArray& gripPoints,AcDbIntArray&  osnapModes,AcDbIntArray&  geomIds) const;
 
  virtual Acad::ErrorStatus   subMoveGripPointsAt( const AcDbIntArray& indices,const AcGeVector3d& offset);
public:
  virtual Acad::ErrorStatus   setColorIndex(Adesk::UInt16 color, Adesk::Boolean doSubents=true);
private:
  //圆心
  AcGePoint3d m_CenterPoint ;
};
2)cpp
//首先是要加入的动态识别一些函数的实现
ACRX_DXF_DEFINE_MEMBERS(CCustomColorCircle, AcDbEntity, AcDb::kDHL_CURRENT,
    AcDb::kMReleaseCurrent, 0, CUSTOMCIRCLE, /*MSG0*/"TSZ");
MAKE_ACDBOPENOBJECT_FUNCTION(CCustomColorCircle) ;
/*
在源文件中我们来分析每个函数的实现,以及它们在什么情况下被调用
*/
/*
这里定义一个版本号,随着版本的提升,数据也可能发生变化,有可能增加,有可能减少,所以针对不同的版本
必须用不同的方式进行处理(序列化与反序列化也是必须的)
*/
#define VERSION 1
//我们知道针对非自定义实体,不是每一种类型的实体,对其设置setColorIndex都能起作用,同理如果你想让你的自定义实体能够通过setColorIndex设置颜色,必须重写这个函数
virtual Acad::ErrorStatus   CCustomColorCircle::setColorIndex(Adesk::UInt16 color, Adesk::Boolean doSubents=true)
{
     return AcDbEntity::setColorIndex( color , doSubents ) ;
}
/*
用途:这个函数就是你自定义实体的几何表现,说白了你要让你自定义的实体以何种面目示人
当调用如下函数时会被调用:所有调用更新绘制的时候
不实现会发生什么:(不实现无任何意义)
*/
virtual Adesk::Boolean      CCustomColorCircle::subWorldDraw(AcGiWorldDraw* mode)
{
     //因为能够通过setColorIndex设置颜色,在这里设置一下
     int colorIndex = this->colorIndex ;
     mode->subEntityTraits().setColor( colorIndex ) ;
     //接下来绘制圆,这里为了简单(偷懒)我们对圆的半径设置一个固定的值
     double dCircleRadius = 200.0 ;
     mode->geometry().circle( m_CenterPoint , dCircleRadius , AcGeVector3d::kZAxis ) ;
     return Adesk::kTrue ;
}
/*
用途:获得一个实体的范围(一个极大点  一个极小点)
当调用如下函数时会被调用:CCustomColorCircle.getGeomExtents
不实现会发生什么:得到的范围无意义(得到的极大点 和极小点的坐标都是没有意义的浮点数)
*/
virtual Acad::ErrorStatus   CCustomColorCircle::subGetGeomExtents(AcDbExtents& extents) const
{
     extents.addPoint( m_CenterPoint + sqrt(2)*200.0*AcGeVector3d(1,1,0) ) ;
     extents.addPoint( m_CenterPoint - sqrt(2)*200.0*AcGeVector3d(1,1,0) ) ;
     return Acad::eOk ;
}
/*
用途:dwgIn:利用图纸初始化数据成员(反序列化)  dwgOut:将当前数据成员写入图纸(序列化)
深入分析:想象一种情况,你建立了一个自定义实体添加到模型空间中,然后保存图纸->关闭图纸,然后再次打开图纸,这时你利用ID打开自定义实体,这时数据成员应该如何初始化?
如果你想通了,就能够明白为什么要这么做,其实就是序列化与反序列化,因为dwg是一个存储在硬盘上的文件
这也就意味着,每一个数据成员都需要需要序列化与反序列化(这是保证你的自定义实体能够使用的基本条件)
在什么时候被调用(不一定每一次都发生):dwgin:save  saveas  wblock insert copy purge
                                 dwgOut:wblock save saveas acdbEntGet
*/
virtual Acad::ErrorStatus CCustomColorCircle::dwgInFields(AcDbDwgFiler* filer)
{
assertWriteEnabled() ;
if ( AcDbEntity::dwgInFields ( filer ) != Acad::eOk )
{
return filer->filerStatus() ;
}
int version ;
filer->readItem(&version);
filer->readItem( &m_CenterPoint ) ;
return filer->filerStatus() ;
}
virtual Acad::ErrorStatus CCustomColorCircle::dwgOutFields(AcDbDwgFiler* filer) const
{
assertReadEnabled() ;
if ( AcDbEntity::dwgOutFields ( filer ) != Acad::eOk )
{
return filer->filerStatus() ;
}
filer->writeItem( VERSION ) ;
filer->writeItem( m_CenterPoint ) ;
return filer->filerStatus() ;
}
/*
dxfin 和dxfOut的意义按着我的理解是为了和dwg保持一致
并且dxf格式也是dwg自始至终都存的东西,并且在构建选择集的时候也很有用
*/
virtual Acad::ErrorStatus CCustomColorCircle::dxfInFields(AcDbDxfFiler* filer)
{
assertWriteEnabled();
if( AcDbEntity::dxfInFields(filer) != Acad::eOk ||
filer->atSubclassData( _T("CCustomColorCircle") ) )
{
return filer->filerStatus() ;
}
//读取版本
     resbuf rb ;
     filer->readItem( &rb ) ;
     int iVersion = rb.resval.rint ;
     //读取圆心
  filer->readItem(&rb) ;
  m_CenterPoint = AcGePoint3d( rb.resval.rpoint[0] , rb.resval.rpoint[1] , 
rb.resval.rpoint[2] ) ;
 
   
}
virtual Acad::ErrorStatus CCustomColorCircle::dxfOutFields(AcDbDxfFiler* filer) const
{
     assertReadEnabled();
     Acad::ErrorStatus es ;
     if ( ( es = AcDbEntity::dxfOutFields(filer)) 
          != Acad::eOk )
     {
          return es ;
     }
    filer->wirteItem( AcDb::kDxfSubClass , _T("CCustomColorCircle") ) ;
    filer->writeInt16( AcDb::kDxfInt16 , VERSION ) ;
    filer->writePoint3d( AcDb::kDxfXCoord , m_CenterPoint ) ;
    return filer->filerStatus() ;
}
/*
用途:选中一下cad提供的实体,例如直线,你会看到三个点(起点 中点  和 终点),选中一个圆,你会看到一个圆心,选择圆弧,会看到三个点(圆弧起点 圆弧中点  圆弧圆心),并且通过拖动这些点能够实现简单的
编辑功能
不实现会发生什么:你选中你添加到空间中的实体,你会发现不会有特征点显示出来,也就无法通过用户交互的方式编辑这个实体(当然可以通过程序实现)
注意:在这里我们多添加几个点到gripPoints中(这个有助于我们分析subMoveGripPointsAt函数)
*/
virtual Acad::ErrorStatus   CCustomColorCircle::subGetGripPoints(AcGePoint3dArray& gripPoints,AcDbIntArray&  osnapModes,AcDbIntArray&  geomIds) const
{
     /*这里添加三个点 添加三个点的顺序很重要
     向gripPoints中填充点,这些点在实体被选中后,会显示出来
     */
     gripPoints.append( m_CenterPoint ) ;
     gripPoints.append( m_CenterPoint + 20 * AcGeVector3d::kXAxis ) ;
     gripPoints.append( m_CenterPoint - 20 * AcGeVector3d::kXAxis ) ;
}
/*
用途:subGetGripPoints只让可以被编辑的点被显示出来,但是真正发挥作用的却是subMoveGripPointsAt
这个函数决定了 当用户拖拽特征点的时候,实体应该如何变化
如果不实现会怎么样:不实现,你就只能看看,无法实现任意的编译功能
*/
virtual Acad::ErrorStatus   CCustomColorCircle::subMoveGripPointsAt( const AcDbIntArray& indices,const AcGeVector3d& offset)
{
     /*
     indices:用户选择的特征点的索引(和你在subGetGripPoints的填充顺序是一致的,从0开始)
     offset(拖拽产生的向量)
     */
     //在这里我们这其实只是一个圆,所以简单调用矩阵变换函数
     if ( indices.length() == 0 || offset.length() == 0 )
          return Acad::eOk ; 
#ifdef _DEBUG
     //输出当前用户拖拽点的索引
     acutPrintf( _T("\n%d") , inDices)
#endif
     return transformBy( AcGeMatrix3d::translation(offset) ) ;//这个函数又会调用subTransformby
}
在dll的入口加入如下两行代码:
     CCustomColorCircle::rxInit() ;
acrxBuildClassHierarchy() ;
总结:
1)适用情况:
     通常情况下,并不建议使用自定义实体,但是有一种情况除外,作为jig的entity实体使用时,这时适用自定义实体再合适不过,而且只需
实现一个函数subWorldDraw(),其余情况(说白了就是实体需要存在dwg文档中时),这里的原则是能不用就不用
2)如何学习:
     我这里也只做了一个基本的介绍,还有很多可以重写的函数,并且能够实现非常强大的功能,那么我们如和才能找到我们需要实现哪一个函数,
这里也有一个技巧,我们发现重载的基本上都会加上一个sub,比如我们对于cad已经提供的实体调用trans,那么在自定义实体中重载的就是sub,
当然这只符合一些编辑功能的重写,其余的要做到两点,最好将你想实现的效果,在cad提供的实体中通过,然后通过编程实现效果
3)给出代码的问题:
     我这是在外网机器上一行一行敲的,难免出现问题
4)进阶:
     在ObjectArx中提供了一个例子 samples/entity/polysamp,这个例子非常好
5)千万不要忘了
CCustomColorCircle::rxInit() ;
acrxBuildClassHierarchy() ;
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

ObjectArx 自定义实体 的相关文章

  • 51单片机上连YL69土壤湿度传感器获取的数据在LCD上显示出来

    要做一个项目 被分配到做DS18B20温度传感与YL69土壤湿度传感器在51单片机上用LCD显示屏显示出来 温度传感模块很简单 网上到处都是资料 但是YL69的资料就很少了 特别还是在51单片机上实现 其实懂了原理也还是简单 将传感器的AO
  • 高并发+海量数据下如何实现系统解耦?【上】

    V xin ruyuanhadeng获得600 页原创精品文章汇总PDF 一 写在前面 之前更新过一个 亿级流量系统架构 系列 主要讲述了一个大规模商家数据平台的如下几个方面 如何承载百亿级数据存储 如何设计高容错的分布式架构 如何设计承载
  • 拓端tecdat

    最近我们被客户要求撰写关于偏最小二乘法 PLS 回归的研究报告 包括一些图形和统计输出 本文建立偏最小二乘法 PLS 回归 PLSR 模型 以及预测性能评估 为了建立一个可靠的模型 我们还实现了一些常用的离群点检测和变量选择方法 可以去除潜
  • 全国程序员收入大调查,粒度到省

    2019年五一假期 我没休息 而是统计某招聘网站了全国的程序员工资 总体统计 2019年4月全国招收程序员302303人 2019年4月全国程序员平均工资12807元 工资中位数11500元 其中95 的人的工资介于3750元到32500元
  • (python算法)LeetCode-版本号比较

    第一次笔试 发挥的很糟糕 基础不好是硬伤 碰到了版本号比较这个问题 回来后搜了下 发现在LeetCode里有 正好再仔细研究下 以下是原题 比较两个版本号 version1 和 version2 如果 version1 gt version

随机推荐

  • python怎么一步步调试_PyCharm入门第一步(二)——调试第一个Python应用程序

    第2步 调试您的第一个Python应用程序 找出问题的根源 PyCharm报告运行时错误 a ZeroDivisionError 深入研究一下代码 找出问题所在 这里可以使用PyCharm调试器来查看代码中发生了什么 要开始调试 您必须先设
  • 【珍藏版】 2012Java开发工程师必备精品资料(115个)

    Java应用广泛 涉及个人PC 数据中心 游戏控制台 科学超级计算机 移动电话和互联网等领域 同时拥有全球最大的开发者专业社群 小弟精心整理了115个精品资料 包括11个Java开发专题和104个热门资源 网上的资料众多 参差不齐 然而这批
  • PHPBONE使用问题集--.Net直接POST数据被过滤

    当 NET用POST发送数据到服务端时 发现 加号全被过滤成空格了 以为是PHPBONE的问题 查了半天代码也没发现哪有异常 但是以前也遇到过 也的确是处理过 只是不记得是怎么处理的了 无耐翻出以前的程序查找了一番 结果发现是编码问题 把数
  • 2021-07-18

    JQuery之DOM操作 1 创建节点及结点属性 1 DOM创建节点及结点属性 创建流程比较简单 大体如下 创建节点 常见的 元素 属性和文本 添加节点的一些属性 加入到文档中 流程中涉及的一点方法 创建元素 document create
  • 哲学家问题(死锁问题)

    1 问题描述 有五个哲学家绕着圆桌坐 每个哲学家面前有一盘面 两人之间有一支筷子 这样每个哲学家左右各有一支筷子 哲学家有2个状态 思考或者拿起筷子吃饭 如果哲学家拿到一只筷子 不能吃饭 直到拿到2只才能吃饭 并且一次只能拿起身边的一支筷子
  • git从某个分支创建新分支

    如题 记录一下从某个分支创建新分支的方法 如从dev分支创建一个test分支 第一步 切换到你指定的分支 如我要从dev上拉一个分支 代码一模一样 git checkout dev 第二步 拉取dev的最新代码 git pull 第三步 在
  • Android Bitmap加载内存占用彻底分析

    背景 在某个版本应用上线后 偶然测得首页占用的内存非常的大而且一直不能回收掉 经过一轮的排查后最终确定是3张图片引起的 当时每张图片占用了将近20m内存 当时紧急处理好后还一直惦记着此事 后来对Android加载Bitmap的内存占用作了彻
  • Android系统源代码的下载与编译

    http www jianshu com p aeaceda41798
  • UVa 12955 Factorial

    Problem uva onlinejudge org index php option com onlinejudge Itemid 8 page show problem problem 4834 开始想多了 想着不能简单贪心 要用dp
  • C# Task异步编程

    Task任务用法 Task用的是线程池 线程池的线程数量的有上限的 这个可以通过ThreadPool修改 我们经常会用到task run new task 和task factory startnew方法来创建任务 Task Factory
  • 检查HDFS块状态

    hadoop集群运行过程中 节点的块状态或者上下线节点时集群都会受影响 如何查看当前的hdfs的块的状态 hadoop1 x时候的命令 hadoop2 x也可使用 hadoop fsck 在hadoop2 0之后 可以使用新命令 hdfs
  • 关于 JavaScript 中的 Promises

    在 JavaScript 中 Promise 是一个对象 它表示一个可能还不可用 但会在未来解决的值 Promises 用于处理异步操作 例如发出网络请求或访问数据库 其中结果不是立即可用的 如果你准备好了 我想开始我们的冒险 承诺如何运作
  • vc2010中开始执行不调试灰的_Intellij IDEA调试功能使用总结

    这段时间一直在使用Intellij IDEA 今天把调试区工具的使用方法记录于此 先编译好要调试的程序 1 设置断点 选定要设置断点的代码行 在行号的区域后面单击鼠标左键即可 2 开启调试会话 点击红色箭头指向的小虫子 开始进入调试 IDE
  • 快速傅里叶变换(FFT)

    前言 在学习FFT过程中看了很多博客 但发现在看博客的时候博客上的内容大多都晦涩难懂 于是乎想自己写一篇博客来记录一下自己学习的心得体会 知其源 先来讲讲FFT的起源 快速傅里叶变换是1965年由J W 库利和T W 图基提出的 采用这种算
  • 微服务项目之JVM Thread线程数飙升

    查找步骤如下 1 找到使用的jdk 在下面找到bin目录 之后双击bin目录下jvisualvm exe 2 启动本地项目 按照上图就可以监控线程数变化 如果项目部署在远程服务器上 那么则需要添加远程监控 选中远程 gt 右键添加远程主机
  • 目标检测跟踪算法--传统方法

    第一阶段 目标跟踪分为两个部分 一个是对指定目标寻找可以跟踪的特征 常用的有颜色 轮廓 特征点 轨迹等 另一个是对目标特征进行跟踪 1 静态背景 1 背景差 对背景的光照变化 噪声干扰以及周期性运动等进行建模 通过当前帧减去背景图来捕获运动
  • python使用hash256加密验证字符串

    这篇博客使用hash256加密一个固定的字符串 同一个字符串有固定的hash256 因此可以通过验证hash值 判断字符串是否发生变化 import hashlib def get hash256 data str 对data加密 hash
  • USB audio调试

    androidstudio打印的信息有如下 07 12 08 27 17 660 2284 2284 I AudioFlinger loadHwModule Loaded a2dp audio interface from A2DP Aud
  • Linux Foundation发布面向IoT的实时OS开发项目「Zephyr」

    本文翻译至 http japan zdnet com article 35078243 物联网 IoT 领域新的玩家出场了 这就是Linux Foundation的 Zephyr Project 本文考察该项目给开发者和消费者双方带来利益的
  • ObjectArx 自定义实体

    1 arx文档中规定的必须重写的几个函数 AcDbObject virtual Acad ErrorStatus dwgInFields AcDbDwgFiler filer virtual Acad ErrorStatus dwgOutF