当前位置: 代码迷 >> 综合 >> AutoCAD.Net开发问题之:层表事件的响应
  详细解决方案

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

热度:74   发布时间:2023-12-22 07:39:12.0

起因


最近想仿照 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 自带图层管理器。

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