AutoCAD.Net开发问题之:层表事件的响应

起因


最近想仿照 ArcGIS 的内容列表窗口做个 CAD 的图层管理器,方便图层数据管理。主要用来实现一些图层常用操作:缩放图层、打开属性表、加载外部数据、导出数据等。

为了保证和系统图层的一致联动,图层管理器要响应一大堆事件:软件初始化加载、文档焦点切换、图层添加删除、层表修改之类,主要问题就在层表修改这里。

问题描述


CAD用层表(LayerTable)记录图层,遍历层表可以获得层表记录(LayerTableRecord),层表记录包含图层的开关、冻结、锁定、颜色等各种信息。

在做到层表事件响应的时候,遇到一些奇怪的问题,没得到直接解决,先记录下:

  1. 新建图层:对层表订阅了 Modified 事件后,通过 CAD 自带的图层管理器添加图层,Modified事件得到响应,但最新添加的图层并未更新到层表中。
  2. 删除图层:通过 CAD 图层管理器删除图层时,Modified 事件并未得到响应。

测试代码如下:

        public void Initialize()
        {
            Database db = Autodesk.AutoCAD.DatabaseServices.HostApplicationServices.WorkingDatabase;
            using (Transaction m = db.TransactionManager.StartTransaction())
            {
                LayerTable lyrTable = m.GetObject(db.LayerTableId, OpenMode.ForRead) as LayerTable;
                lyrTable.Modified += LyrTable_Modified;
                m.Commit();
            }
        }

        private void LyrTable_Modified(object sender, EventArgs e)
        {
            LayerTable lyrTable = sender as LayerTable;
            Application.DocumentManager.MdiActiveDocument.Editor.WriteMessage(
                "当前图层数量:" + GetLayerCount(lyrTable) + Environment.NewLine);
        }

        private int GetLayerCount(LayerTable lyrTable)
        {
            int count = 0;
            foreach (ObjectId id in lyrTable)
                count++;
            return count;
        }

这是执行效果:

在这里插入图片描述
可以看到如上两个问题,不晓得是不是对 Modified 事件理解有误,还是代码哪儿不对,如果有遇到类似问题并得到解决的大大,望不吝赐教。

替代解决方案


以上问题找不到原因,于是找了个替代解决方案:直接订阅 Dababase 的 Object 变更事件,在事件中判断 Object 类型,如果是 LayerTableRecord 则对图层进行修改

主要代码:

        /// <summary>
        /// 订阅事件
        /// </summary>
        /// <param name="db"></param>
        private void AddEvent(Database db)
        {
            if (db == null) return;
            db.ObjectErased -= WorkingDatabaseOnObjectErased;
            db.ObjectErased += WorkingDatabaseOnObjectErased;

            db.ObjectAppended -= WorkingDatabaseOnObjectAppended;
            db.ObjectAppended += WorkingDatabaseOnObjectAppended;

            db.ObjectModified -= WorkingDatabaseOnObjectModified;
            db.ObjectModified += WorkingDatabaseOnObjectModified;

            db.ObjectOpenedForModify -= WorkingDatabaseOnObjectOpenedForModify;
            db.ObjectOpenedForModify += WorkingDatabaseOnObjectOpenedForModify;
        }
        
        private void WorkingDatabaseOnObjectModified(object sender, ObjectEventArgs e)
        {
            LayerTableRecord ltr = e.DBObject as LayerTableRecord;
            if (ltr == null) return;
            //do something
        }

运行:

在这里插入图片描述
左边是自定义图层管理器,右边是 CAD 自带图层管理器。

这样做虽然可以实现目的,但对全库对象订阅,修改任一数据库对象都会触发那一大堆事件,不是好的解决方案,暂且记录在此。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值