C# 是否可以使用Autodesk.AutoCAD.Interop在AutoCAD中编辑块属性?
我开发了一个外部WPF应用程序来生成c#中的图形。我已经能够使用Autodesk.AutoCAD.Interop绘制、标注、添加块以及应用程序所需的所有其他内容,但是我似乎无法填充标题栏或生成明细表 我看到的所有示例都基于一种机制,该机制要求应用程序在AutoCAD中作为插件运行。事实是,使用ModelSpace.InsertLine插入一行或两行代码,现在至少需要8行代码 是否有使用Autodesk.AutoCAD.Interop实现此功能的方法?或者有没有一种方法可以将互操作与可以从外部exe调用的插件结合使用 任何关于这方面的建议都将不胜感激 谢谢 编辑 举例说明:C# 是否可以使用Autodesk.AutoCAD.Interop在AutoCAD中编辑块属性?,c#,.net,autocad,C#,.net,Autocad,我开发了一个外部WPF应用程序来生成c#中的图形。我已经能够使用Autodesk.AutoCAD.Interop绘制、标注、添加块以及应用程序所需的所有其他内容,但是我似乎无法填充标题栏或生成明细表 我看到的所有示例都基于一种机制,该机制要求应用程序在AutoCAD中作为插件运行。事实是,使用ModelSpace.InsertLine插入一行或两行代码,现在至少需要8行代码 是否有使用Autodesk.AutoCAD.Interop实现此功能的方法?或者有没有一种方法可以将互操作与可以从外部ex
// before - Draw Line with Autodesk.AutoCAD.Interop
private static AcadLine DrawLine(double[] startPoint, double[] endPoint)
{
AcadLine line = ThisDrawing.ModelSpace.AddLine(startPoint, endPoint);
return line;
}
// Now - Draw line with Autodesk.AutoCAD.Runtime
[CommandMethod("DrawLine")]
public static Line DrawLine(Coordinate start, Coordinate end)
{
// Get the current document and database
// Get the current document and database
Document acDoc = Application.DocumentManager.MdiActiveDocument;
Database acCurDb = acDoc.Database;
// Start a transaction
using (Transaction acTrans = acCurDb.TransactionManager.StartTransaction())
{
// Open the Block table for read
BlockTable acBlkTbl;
acBlkTbl = acTrans.GetObject(acCurDb.BlockTableId, OpenMode.ForRead) as BlockTable;
// Open the Block table record Model space for write
BlockTableRecord acBlkTblRec;
acBlkTblRec = acTrans.GetObject(acBlkTbl[BlockTableRecord.ModelSpace], OpenMode.ForWrite) as BlockTableRecord;
// Create a line that starts at 5,5 and ends at 12,3
Line acLine = new Line(start.Point3d, end.Point3d);
acLine.SetDatabaseDefaults();
// Add the new object to the block table record and the transaction
acBlkTblRec.AppendEntity(acLine);
acTrans.AddNewlyCreatedDBObject(acLine, true);
// Save the new object to the database
acTrans.Commit();
return acLine;
}
}
是的,你完全可以把这两种方法结合起来
[CommandMethod("EditBlockAtt")]
public void EditBlockAtt()
{
var acDb = HostApplicationServices.WorkingDatabase;
var acEd = AcadApplication.DocumentManager.MdiActiveDocument.Editor;
var blockNamePrompt = acEd.GetString(Environment.NewLine + "Enter block name: ");
if (blockNamePrompt.Status != PromptStatus.OK) return;
var blockName = blockNamePrompt.StringResult;
var attNamePrompt = acEd.GetString(Environment.NewLine + "Enter attribute name: ");
if (attNamePrompt.Status != PromptStatus.OK) return;
var attName = attNamePrompt.StringResult;
var acPo = new PromptStringOptions(Environment.NewLine + "Enter new attribute value: "){ AllowSpaces = true };
var newValuePrompt = acEd.GetString(acPo);
if (newValuePrompt.Status != PromptStatus.OK) return;
var newValue = newValuePrompt.StringResult;
using (var acTrans = acDb.TransactionManager.StartTransaction())
{
var acBlockTable = acTrans.GetObject(acDb.BlockTableId, OpenMode.ForRead) as BlockTable;
if (acBlockTable == null) return;
var acBlockTableRecord = acTrans.GetObject(acBlockTable[BlockTableRecord.ModelSpace], OpenMode.ForRead) as BlockTableRecord;
if (acBlockTableRecord == null) return;
foreach (var blkId in acBlockTableRecord)
{
var acBlock = acTrans.GetObject(blkId, OpenMode.ForRead) as BlockReference;
if (acBlock == null) continue;
if (!acBlock.Name.Equals(blockName, StringComparison.CurrentCultureIgnoreCase)) continue;
foreach (ObjectId attId in acBlock.AttributeCollection)
{
var acAtt = acTrans.GetObject(attId, OpenMode.ForRead) as AttributeReference;
if (acAtt == null) continue;
if (!acAtt.Tag.Equals(attName, StringComparison.CurrentCultureIgnoreCase)) continue;
acAtt.UpgradeOpen();
acAtt.TextString = newValue;
}
}
acTrans.Commit();
}
}
然后从interop AcadApplication中,网络加载dll,并以以下格式从命令行调用该方法:
(Command "EditBlockAtt" "BlockName" "AttributeName" "NewValue")
但是,如果您希望实现纯互操作,这可能会满足您的需要,因为您在运行时有一个AcadDocument对象:
foreach (AcadEntity ent in acadDoc.ModelSpace)
{
var block = ent as AcadBlockReference;
if (block == null) continue;
{
if (!block.Name.Equals("BlockName", StringComparison.CurrentCultureIgnoreCase)) continue;
var atts = block.GetAttributes() as object[];
if (atts == null) continue;
foreach (var attribute in atts.OfType<AcadAttributeReference>()
.Where(attribute => attribute.TagString.Equals("AttributeName",
StringComparison.CurrentCultureIgnoreCase)))
{
attribute.TextString = "New Value";
}
}
}
foreach(acadDoc.ModelSpace中的AcadEntity ent)
{
var block=ent作为AcadBlockReference;
如果(block==null)继续;
{
如果(!block.Name.Equals(“BlockName”,StringComparison.CurrentCultureIgnoreCase))继续;
var atts=block.GetAttributes()作为对象[];
如果(atts==null)继续;
foreach(atts.OfType()中的var属性)
.Where(attribute=>attribute.TagString.Equals(“AttributeName”),
StringComparison.CurrentCultureIgnoreCase)))
{
attribute.TextString=“新值”;
}
}
}
另请注意,这是在使用AutoCAD 2012互操作库。YMMV.值得注意的是,尽管使用AcMdg.dll和AcDbMdg.dll的AutoCAD内部开发通常更为复杂,但与互操作的对应项相比,它的效率更高。@reckface-您能告诉我如何在不显式打开的情况下向AutoCAD发送命令吗。可以通过纯.Net代码完成吗?我有一个包装器类,适当地称为CAD。我调用CAD.Setup(),它将创建一个新的AutoCAD.Application(如果尚未创建),然后调用app.Documents.Add或create,然后在CAD类中将所有调用包装到文档中。这是一个纯.Net exe安装和ClickOnce部署。必须等到我面前有一个IDE时,才能扩展我的答案以包括interop方法。如果您发现在较大的图形上这需要一段时间,那么您可能需要先过滤一个选择,使其只包含块,然后在其中进行迭代,而不是遍历整个模型空间。谢谢。我正在考虑实现新的acMdg.dll.net引用,但截止日期限制意味着我必须在投入更多时间更好地理解接口之前,现在就制作一些东西。昨天我将所有内容都剥离到acMdg.dll,发现它无法独立运行,我必须快速返回到互操作!再次感谢,我将努力将这两种方法结合起来。@Parrish-请告诉我如何在不明确打开AutoCad的情况下向AutoCad发送命令。可以通过纯.Net代码完成吗?@RohitVats,对AutoCAD的进程内.Net调用假定应用程序正在运行。但是,可以通过互操作实例化AutoCAD,然后向新运行的实例发送进程内命令。你的目标到底是什么?我正在编写一个软件,它将加载dwg文件,从中选择一些块,修改它(设置一些属性等),分解它并保存在新的dwg文件中。但现在我正在努力从进程本身打开AutoCAD。它总是失败,但有一些COM异常:(我已经尝试了这里提到的两种方法,但没有一种适合我。(通过编组尝试,也尝试创建AcadApplication实例)