21/10/2015
Как объединить отрезки и дуги в полилинию?
Эта статья навеяна темой на форуме Полилинии и дуги в одну полилинию
Начиная с AutoCAD 2011 для этой цели можно воспользоваться методами Entity.JoinEntity и Entity.JoinEntities из AutoCAD .NET API, а также AcDbJoinEntityPE::joinEntity и AcDbJoinEntityPE::joinEntities из ObjectARX. Интереснее показать как можно это реализовать самостоятельно в .NET:
Код - C#: [Выделить]
- using System;
- using System.Collections;
- using Autodesk.AutoCAD.Runtime;
- using Autodesk.AutoCAD.DatabaseServices;
- using Autodesk.AutoCAD.ApplicationServices;
- using Autodesk.AutoCAD.EditorInput;
- using Autodesk.AutoCAD.Geometry;
- #pragma warning disable 0618
- [assembly: CommandClass(typeof(Rivilis.Contour))]
- namespace Rivilis
- {
- public class Contour
- {
- [CommandMethod("MakePline")]
- public void MakePline()
- {
- Document doc = Application.DocumentManager.MdiActiveDocument;
- Editor ed = doc.Editor;
- Database db = doc.Database;
- // Отбираем только отрезки и дуги
- SelectionFilter sf =
- new SelectionFilter(new TypedValue[] { new TypedValue((int)DxfCode.Start,"LINE,ARC") });
- PromptSelectionResult rs = ed.GetSelection(sf);
- if (rs.Status != PromptStatus.OK) return;
- ObjectIdCollection ids = new ObjectIdCollection(rs.Value.GetObjectIds());
- while (ids.Count > 0) {
- using (Polyline p = MakeJoinedPoly(ref ids)) {
- if (p != null) {
- using (BlockTableRecord btr = db.CurrentSpaceId.Open(OpenMode.ForWrite) as BlockTableRecord) {
- btr.AppendEntity(p);
- }
- }
- else {
- ed.WriteMessage("\nОшибка во входных данных!");
- return;
- }
- }
- }
- }
- public static Polyline MakeJoinedPoly(ref ObjectIdCollection ids)
- {
- if (ids.Count == 0) return null;
- // Создаём полилинию
- Polyline p = new Polyline();
- p.SetDatabaseDefaults();
- ObjectId idOwn = ObjectId.Null;
- ObjectId idFirst = ids[0];
- Point3d nextPt = Point3d.Origin;
- Point3d prevPt = Point3d.Origin;
- // Добавляем первые две вершины из первого выбранного примитива
- using (Curve c = idFirst.Open(OpenMode.ForRead) as Curve) {
- p.AddVertexAt(0, ToPoint2d(c.StartPoint), BulgeFromArc(c, false), 0, 0);
- p.AddVertexAt(1, ToPoint2d(c.EndPoint), 0, 0, 0);
- nextPt = c.EndPoint;
- prevPt = c.StartPoint;
- idOwn = c.OwnerId;
- }
- ids.Remove(idFirst);
- int prevCnt = ids.Count + 1;
- while (ids.Count > 0 && ids.Count < prevCnt)
- {
- prevCnt = ids.Count;
- foreach (ObjectId id in ids) {
- using (Curve cv = id.Open(OpenMode.ForRead) as Curve) {
- if (cv.StartPoint == nextPt || cv.EndPoint == nextPt) {
- double bulge = BulgeFromArc(cv, cv.EndPoint == nextPt);
- p.SetBulgeAt(p.NumberOfVertices - 1, bulge);
- if (cv.StartPoint == nextPt)
- nextPt = cv.EndPoint;
- else
- nextPt = cv.StartPoint;
- p.AddVertexAt(p.NumberOfVertices, ToPoint2d(nextPt), 0, 0, 0);
- ids.Remove(id);
- break;
- } else if (cv.StartPoint == prevPt || cv.EndPoint == prevPt) {
- double bulge = BulgeFromArc(cv, cv.StartPoint == prevPt);
- if (cv.StartPoint == prevPt)
- prevPt = cv.EndPoint;
- else
- prevPt = cv.StartPoint;
- p.AddVertexAt(0, ToPoint2d(prevPt), bulge, 0, 0);
- ids.Remove(id);
- break;
- }
- }
- }
- }
- if (p.NumberOfVertices == 0)
- return null;
- else
- return p;
- }
- // Функция возвращает кривизну дуги (bulge) или 0.0
- static double BulgeFromArc(Curve c, bool clockwise)
- {
- double bulge = 0.0;
- Arc a = c as Arc;
- if (a == null) return bulge;
- double newStart = (a.StartAngle > a.EndAngle) ?
- (a.StartAngle - 8 * Math.Atan(1)) : (a.StartAngle);
- bulge = Math.Tan((a.EndAngle - newStart) / 4);
- if (clockwise) bulge = -bulge;
- return bulge;
- }
- // Функция преобразует Point3d в Point2d (отбрасывает Z)
- static Point2d ToPoint2d(Point3d p) {
- return new Point2d(p.X, p.Y);
- }
- }
- }
В ObjectARX:
Код - C++: [Выделить]
- //-----------------------------------------------------------------------------
- //----- acrxEntryPoint.cpp
- //-----------------------------------------------------------------------------
- #include "StdAfx.h"
- #include "resource.h"
- //-----------------------------------------------------------------------------
- #define szRDS _RXST("")
- //-----------------------------------------------------------------------------
- //----- ObjectARX EntryPoint
- class CJoinToPlineApp : public AcRxArxApp {
- public:
- CJoinToPlineApp () : AcRxArxApp () {}
- virtual AcRx::AppRetCode On_kInitAppMsg (void *pkt) {
- AcRx::AppRetCode retCode = AcRxArxApp::On_kInitAppMsg (pkt) ;
- return (retCode) ;
- }
- virtual AcRx::AppRetCode On_kUnloadAppMsg (void *pkt) {
- AcRx::AppRetCode retCode = AcRxArxApp::On_kUnloadAppMsg (pkt) ;
- return (retCode) ;
- }
- virtual void RegisterServerComponents () {
- }
- static void JoinToPlineJoinToPline () {
- // Фильтр для выбора отрезков (LINE) и дуг (ARC)
- ACHAR* promptPtrs[] = {
- _T("\nВыберите отрезки и дуги для объединения в полилинии (ENTER - завершение): "),
- _T("\nУдалите линии из набора: ")
- };
- ads_name ss;
- resbuf *rbFilter = acutBuildList(RTDXF0,_T("LINE,ARC"),RTNONE);
- int rc = acedSSGet(_T(":$"),promptPtrs,NULL,rbFilter,ss);
- acutRelRb(rbFilter);
- if (rc != RTNORM) return;
- AcDbObjectIdArray ids;
- if (ObjectIdArrayFromSelSet(ss, ids) != Acad::eOk) return;
- while (ids.length() > 0)
- {
- AcDbEntity *p = MakeJonedPoly(ids);
- if (p) {
- AcDbBlockTableRecordPointer pSpace(acdbCurDwg()->currentSpaceId(), AcDb::kForWrite);
- if (pSpace.openStatus() != Acad::eOk) return;
- pSpace->appendAcDbEntity(p);
- p->close();
- } else {
- acutPrintf(_T("\nОшибка во входных данных!"));
- return;
- }
- }
- }
- /// <summary>
- /// Получение AcDbObjectIdArray из SelectionSet
- /// </summary>
- /// <param name="sset">SelectionSet</param>
- /// <param name="ids">AcDbObjectIdArray</param>
- /// <returns></returns>
- static Acad::ErrorStatus ObjectIdArrayFromSelSet(ads_name sset, AcDbObjectIdArray &ids)
- {
- Acad::ErrorStatus es = Acad::eOk;
- long nset = -1;
- if (acedSSLength(sset,&nset) != RTNORM) return Acad::eAmbiguousInput;
- ids.setLogicalLength(nset);
- ads_name en;
- AcDbObjectId id;
- for (long i=0; i < nset; i++) {
- if (acedSSName(sset,i,en) == RTNORM) {
- if ((es = acdbGetObjectId(id,en)) != Acad::eOk) return es;
- ids[i] = id;
- }
- }
- return Acad::eOk;
- }
- /// <summary>
- /// Создаём полилинию из переданных отрезков и дуг.
- /// Их идентификаторы удаляются из переданного массива
- /// </summary>
- /// <param name="ids">Массив идентификаторов</param>
- /// <param name="FUZZ">Точность определения расстояния между точками</param>
- /// <returns></returns>
- static AcDbPolyline* MakeJonedPoly(
- AcDbObjectIdArray &ids,
- double FUZZ = AcGeContext::gTol.equalPoint())
- {
- AcDbPolyline *p = new AcDbPolyline();
- p->setDatabaseDefaults();
- AcDbObjectId idFirst = ids[0];
- AcGePoint3d nextPt = AcGePoint3d::kOrigin;
- AcGePoint3d prevPt = AcGePoint3d::kOrigin;
- AcDbObjectPointer<AcDbCurve> c(idFirst,AcDb::kForRead);
- if (c.openStatus() == Acad::eOk) {
- AcGePoint3d ptStart, ptEnd;
- c->getStartPoint(ptStart); c->getEndPoint(ptEnd);
- p->addVertexAt(0, asPnt2d(asDblArray(ptStart)), BulgeFromArc(c, false), 0, 0);
- p->addVertexAt(1, asPnt2d(asDblArray(ptEnd)), 0, 0, 0);
- nextPt = ptEnd;
- prevPt = ptStart;
- }
- ids.remove(idFirst);
- int prevCnt = ids.length() + 1;
- while (ids.length() > 0 && ids.length() < prevCnt)
- {
- prevCnt = ids.length();
- for (int i = 0; i < ids.length(); i++) {
- AcDbObjectId id = ids[i];
- AcDbObjectPointer<AcDbCurve> cv(id,AcDb::kForRead);
- if (cv.openStatus() == Acad::eOk) {
- AcGePoint3d ptStart, ptEnd;
- cv->getStartPoint(ptStart); cv->getEndPoint(ptEnd);
- if (ptStart.distanceTo(nextPt) < FUZZ || ptEnd.distanceTo(nextPt) < FUZZ) {
- double bulge = BulgeFromArc(cv, ptEnd.distanceTo(nextPt) < FUZZ);
- p->setBulgeAt(p->numVerts() - 1, bulge);
- if (ptStart.distanceTo(nextPt) < FUZZ)
- nextPt = ptEnd;
- else
- nextPt = ptStart;
- p->addVertexAt(p->numVerts(), asPnt2d(asDblArray(nextPt)), 0, 0, 0);
- ids.remove(id);
- break;
- } else if (ptStart.distanceTo(prevPt) < FUZZ || ptEnd.distanceTo(prevPt) < FUZZ) {
- double bulge = BulgeFromArc(cv, ptStart.distanceTo(prevPt) < FUZZ);
- if (ptStart.distanceTo(prevPt) < FUZZ)
- prevPt = ptEnd;
- else
- prevPt = ptStart;
- p->addVertexAt(0, asPnt2d(asDblArray(prevPt)), bulge, 0, 0);
- ids.remove(id);
- break;
- }
- }
- }
- }
- if (p->numVerts() == 0) {
- delete p; return NULL;
- } else {
- return p;
- }
- }
- /// <summary>
- /// Получение кривизны (bulge) для кривой
- /// </summary>
- /// <param name="c">Указатель на кривую</param>
- /// <param name="clockwise">по часовой ли стрелке.</param>
- /// <returns></returns>
- static double BulgeFromArc(AcDbCurve *c, bool clockwise)
- {
- double bulge = 0.0;
- AcDbArc *a = AcDbArc::cast(c);
- if (a == NULL) return bulge;
- double newStart =
- (a->startAngle() > a->endAngle()) ?
- (a->startAngle() - 8 * atan(1.0)) :
- (a->startAngle());
- bulge = tan((a->endAngle() - newStart) / 4.0);
- if (clockwise) bulge = -bulge;
- return bulge;
- }
- } ;
- //-----------------------------------------------------------------------------
- IMPLEMENT_ARX_ENTRYPOINT(CJoinToPlineApp)
- ACED_ARXCOMMAND_ENTRY_AUTO(CJoinToPlineApp, JoinToPline, JoinToPline, JoinToPline, ACRX_CMD_MODAL, NULL)
Автор: Александр Ривилис
Отредактировано 21.10.2015 в 22:57:02
Обсуждение: http://adn-cis.org/forum/index.php?topic=3122
Опубликовано 21.10.2015Отредактировано 21.10.2015 в 22:57:02