实体
图纸所有对象都是从类 McDbObject 继承,该类提供得到对象句柄、ID、对象类型信息等函数。图上可见的对象是从类 McDbEntity 继承,该类提供可见实体的通用属性操作,如线型,图层,文字样式,颜色等函数。
点击 数据库对象 McDbObject、实体对象 McDbEntity 查看详细属性和方法说明。
实体ID和句柄
ID和句柄都用来标识一个对象,ID是一个64位长整型变量,是一个内存地址,访问速度最快,但它每次打开都不一样,如果需要把一个对象的标识存起来,下次又能找到对象,就需要使用句柄,句柄是一个字符串变量,它保证不管什么时候总是不会变化,句柄在DWG图纸中是唯一存在,ID是内存中唯一存在。
我们可以通过 McDbObject 类中提供的 getHandle() 方法获得对象句柄,通过 getObjectID() 方法获得对象ID。
点击 McDbObject.getHandle()、McDbObject.getObjectID() 查看详细属性和方法说明。
// 选择一个目标对象
const getEnt = new MxCADUiPrEntity();
getEnt.setMessage('选择目标对象');
const entVal = await getEnt.go();
if (!entVal.id) return;
// 获取该对象的实体
const ent = entVal.getMcDbEntity();
if(!ent) return;
// 获取对象ID
const entId = ent.getObjectID();
// 获取对象句柄
const sHandle = ent.getHandle();
console.log("对象id", entId);
console.log("对象句柄", sHandle);
实体的类型信息
我们可以通过调用 McDbObject 类中提供的 objectName 属性得到对象的类型名,有了类型名就能判断实体具体是一个什么对象,再调用 dxf0 属性,得到对象的DXF组码的类型名,这个和AutoCAD中的DXF组码是一样,比如直线的类型名为:McDbLine,DXF0组码值: LINE,DXF0组码值可以用来构造集时的类型过滤。
点击 McDbObject.objectName、McDbObject.dxf0 查看详细属性和方法说明。
// 让用户在图上选择一个对象。
const getEnt = new MxCADUiPrEntity();
getEnt.setMessage('选择目标对象');
const entVal = await getEnt.go();
if (!entVal.id) return;
// 获取对象实体
const ent = entVal.getMcDbEntity();
// 得到Dxf0的值。
const sDxfType = ent?.dxf0;
console.log("Dxf0值", sDxfType)
if (ent?.objectName == "McDbLine"){
// ent是一个直线
const line = ent as McDbLine;
// 得到直线开始点
const pt1 = line.startPoint;
// 得到直线结束点
const pt2 = line.endPoint;
console.log(pt1, pt2);
}else if(ent?.objectName == "..."){
// 其他类型
}
可见实体属性
图上可见实体 McDbEntity 的主要属性,有线型,图层,文字样式,颜色,是否可见,显示顺序等。我们可以提取和修改这些属性。
- 获取、设置实体是否可见
// 假设entity为实体对象McDbEntity
const isVisible = entity.visible; // 得到实体是否可见
console.log("实体是否可见", isVisible);
entity.visible = false;// 隐藏实体
entity.visible = true;// 显示实体
- 获取、修改对象颜色
- 获取对象颜色
// 选择目标对象
let getEntity = new MxCADUiPrEntity();
getEntity.setMessage("选择目标对象");
let id = await getEntity.go();
if (!id.isValid()) return;
let entity = id.getMcDbEntity();
if (!entity) return;
// 修改对象颜色
entity.trueColor = new McCmColor(255, 0, 255);
- 修改对象颜色
mxcad 实体对象中有一个 trueColor 属性,用户可以通过在实体中调用该属性来获取实体的颜色,但由于实体对象可能设置了颜色索引值,如随层、随块等并未直接为实体设置颜色,因此直接通过 trueColor 属性获取到的颜色可能与预期不符。为了避免上述问题,用户通过下面的示例代码来获取实体的实际颜色和实际颜色索引值。
import { McCmColor } from "mxcad";
// 假设entity为实体对象McDbEntity
const layerId = entity.layerId;
/**
* 如果目标实体为图块中的实体,则还需要在getColorValue()方法中传入图块ID
* 否则,传入图层ID即可。
* blkId:图块ID
* entity.trueColor.getColorValue(layerId, blkId)
*/
const val = entity.trueColor.getColorValue(layerId);
const hexValue = parseInt(val, 16);
const rgb =[(hexValue >> 16) & 255,(hexValue >> 8) & 255, hexValue & 255]
const color = new McCmColor();
color.setRGB(...rgb);
console.log('实体实际颜色索引值:' + color.colorIndex) ;
console.log('实体实际颜色:' + color);
- 获取、修改显示顺序
- 获取显示顺序
// 假设entity为实体对象McDbEntity
const drawOrder = entity.drawOrder; // 得到实体的显示顺序
console.log("实体的显示顺序", drawOrder);
- 修改显示顺序
在用户绘图时,在默认情况下,重叠对象(例如文字、宽多段线和实体填充多边形)通常按其创建次序显示:新创建的对象显示在现有对象前面。本实例演示了如何控制显示顺序,具体实现代码如下:
let ss = new MxCADSelectionSet();
if (!await ss.userSelect("\n选择对象")) return;
//得到当前图上对象的最大,最小显示顺序.
let minmaxOrder = MxCpp.getCurrentDatabase().currentSpace.getMinMaxDrawOrder();
// 把对象放到最上面。
let lOrder = minmaxOrder.maxDrawOrder + 1;
ss.forEach((id) => {
let ent = id.getMcDbEntity();
if (ent) {
ent.drawOrder = lOrder;
}
})
- 获取、修改对象图层
- 获取对象图层
// 假设entity为实体对象McDbEntity
const layerId = entity.layerId; // 得到实体所在图层ID
console.log("实体所在图层ID", layerId);
const layerName = entity.layer; // 得到图层名称
console.log("实体所在图层名称", layerName);
- 修改对象图层
// 修改实体所在图层
const layerTable = mxcad.getDatabase().getLayerTable();// 拿到当前控件的数据库图层表
const layerName = "测试图层";// 图层名称
if (layerTable.has(layerName)) {
line.layerId = layerTable.get(layerName);
} else {
const layerId = mxcad.addLayer(layerName);
line.layerId = layerId;
}
const layerId = mxcad.addLayer(layerName)
entity.layerId = layerId;
const mxcad = MxCpp.getCurrentMxCAD();
mxcad.updateDisplay();
console.log("实体所在图层名", entity.layer);
扩展数据
用户可给图纸上的对象设置扩展数据,这些扩展数据可以是用户实际需求的专业数据,扩展数据支持字符串、double、int等类型。每个扩展数据都一个扩展数据名称,每个名称下也可能有多个扩展数据,一个实体可以有多个扩展数据名称,这些扩展数据在内存中是一个链表 MxCADResbuf 来存放的,每个链结代表一个数据,每个链结中字符串数据长度不超过200(由DWG图纸格式确定)。
点击 链表 MxCADResbuf 查看详细属性和方法说明。
// 让用户在图上选择一个对象。
const getEnt = new MxCADUiPrEntity();
getEnt.setMessage('选择目标对象');
const entVal = await getEnt.go();
if (!entVal.id) return;
// 获取对象实体
const ent = entVal.getMcDbEntity();
if(!ent) return;
const xData = new MxCADResbuf();
// 设置扩展数据名称 TestApp
xData.AddString("TestApp",1001);
// 设置字符串数据,DXF组码是1000;
xData.AddString("string data", 1000);
// 设置一个 Long数据,DXF组码是1071
xData.AddLong(77677, 1071);
// 把准备好的数据,写到实体上。
ent.setxData(xData);
编辑实体
我们提供了复制、平移、缩放、变换、镜向、旋转等编辑实体的函数。下面以对实体进行复制平移为例,更多的编辑函数请参考 McDbEntity 。
- 复制平移
import { MxCpp, MxCADUiPrEntity, MxCADUiPrPoint, McGeMatrix3d} from "mxcad";
// 复制平移
export async function test_copyMove() {
// 获取画布实例
const mxcad = MxCpp.App.getCurrentMxCAD();
let copyObj = new MxCADUiPrEntity();//选择复制对象
copyObj.setMessage('请选择需要复制的对象')
let copyObj_id = await copyObj.go();
if (copyObj_id === null) return;
let copyEvent: any = await copyObj_id.getMcDbEntity()//复制的实例对象
if (!copyEvent) return;
let getFristPoint = new MxCADUiPrPoint();
getFristPoint.setMessage('请设定复制基点')
let fristPoint: any = await getFristPoint.go();//基点
if (!fristPoint) return;
let getNextPoint = new MxCADUiPrPoint();
getNextPoint.setMessage('请设置移动长度和方向');
// 动态绘制
getNextPoint.setUserDraw((pt, pw) => {
let evrnt_clone = copyEvent.clone();// 复制实体
let matrix = new McGeMatrix3d();// 创建一个矩阵
matrix.setToTranslation(pt.sub(fristPoint));//设置移动矩阵
evrnt_clone.transformBy(matrix);// 变换实体
pw.drawMcDbEntity(evrnt_clone);// 动态绘制平移后的实体
let pl = new McDbPolyline();// 创建多段线对象
pl.addVertexAt(fristPoint);
pl.addVertexAt(pt);
pw.drawMcDbEntity(pl);// 绘制平移路径
});
let nextPoint = await getNextPoint.go();
if (!nextPoint) return;
// 平移变换
let matrix = new McGeMatrix3d();// 创建一个矩阵
let evrnt_clone = copyEvent.clone();
matrix.clone();
matrix.setToTranslation(nextPoint.sub(fristPoint));//设置移动矩阵
evrnt_clone.transformBy(matrix);// 变换实体
mxcad.drawEntity(evrnt_clone);// 绘制平移后的实体
}
删除实体
我们可以直接通过实体对象或实体对象id调用 McDbEntity 类中的erase() 方法删除实体。
点击McDbEntity.erase() 查看详细属性和方法说明。
// 选择目标对象
let getEntity = new MxCADUiPrEntity();
getEntity.setMessage("选择目标对象");
let id = await getEntity.go();
if (!id.isValid()) return;
id.erase();
// 选择目标对象
let getEntity = new MxCADUiPrEntity();
getEntity.setMessage("选择目标对象");
let id = await getEntity.go();
if (!id.isValid()) return;
// 获取对象实体
const ent = id.getMcDbEntity();
ent.erase();
得到实体坐标
不同的实体它的坐标属性不一样,比如直线就是开始点,和结束点属性,圆的就是中心点坐标和半径属性。所以要得实体的坐标,需要判断对象类型,然后再转换成具体的对象,再读取它的坐标。我们可以通过调用 McDbEntity 类中的 getBoundingBox() 方法得到对象最小外包矩形框坐标。
点击 McDbEntity.getBoundingBox() 查看详细属性和方法说明。
// 让用户在图上选择一个对象。
const getEnt = new MxCADUiPrEntity();
getEnt.setMessage('选择目标对象');
const entVal = await getEnt.go();
if (!entVal.id) return;
// 获取对象实体
const ent = entVal.getMcDbEntity();
if(!ent) return;
// 得到实体的包围盒
const {ret, minPt, maxPt} = ent.getBoundingBox();
if(ret){
// 将显示范围设置为该包围盒的大小
MxCpp.getCurrentMxCAD().zoomW(minPt, maxPt);
}