简单模拟多段线绘制Pline命令过程的撤销功能

简单模拟多段线绘制Pline命令过程的撤销功能

这是关于多段线pline命令中的撤销功能的简单代码

代码使用了事务管理器来执行增加对象和撤销处理.

代码

		/// <summary>
		/// 简单模拟多段线绘制过程的撤销功能.
		/// code by edata 2020-3-11 
		/// </summary>
		/// <remarks>简单模拟多段线绘制过程的_u撤销功能.仅考虑了UCS问题</remarks>
		//设置坐标系转换缓冲区链表
		resbuf ucs,wcs;
		ucs.restype=RTSHORT;
		ucs.resval.rint=1;
		wcs.restype=RTSHORT;
		wcs.resval.rint=0;

		ads_point pt;
		ads_point ptWcs;
		if (RTNORM != acedGetPoint(NULL,_T("\n 选择起点: "),pt))
		{
			return;
		}
		acedTrans(pt,&ucs,&wcs,0,ptWcs);
		AcGePoint3d p1=asPnt3d(ptWcs);
		AcGePoint3d ptStart=p1;
		AcDbPolyline *pPoly=NULL;
		//开启事务管理器
		AcTransaction *pTrans =actrTransactionManager->startTransaction();
		while (true)
		{
			CString strPrompt=_T("\n 选择下一个点[(U)撤销]: ");
			CString strKword=_T("Undo _ u");
			if (pPoly!=NULL)
			{
				int nCount =pPoly->numVerts();
				if (nCount>1)
				{
					strPrompt=_T("\n 选择下一个点[(U)撤销/(C)闭合]: ");
					strKword=_T("Undo Close _ u c");	
										
				}
			}
			
			//设置使用Undo关键字,该值允许使用ctrl+z撤销
			acedInitGet(NULL,strKword);
			int nRet=acedGetPoint(pt,strPrompt,pt);
			if (RTKWORD == nRet)
			{
				ACHAR szKword[132];
				if (RTNORM != acedGetInput(szKword))
				{
					continue;
				}
				if (_tcscmp(szKword,_T("u")) == 0)
				{
					if (pPoly != NULL)
					{
						int nCount = pPoly->numVerts();
						if (nCount>0)
						{
							//此处使用removeVertexAt不能移除仅剩的一个点,后面对1个点的线取消事务
							Acad::ErrorStatus es= pPoly->removeVertexAt(nCount-1);	
							//重新生成
							pPoly->draw();
							if (nCount>1)
							{
								pPoly->getEndPoint(p1);
								pt[X] = p1.x;
								pt[Y] = p1.y;
								pt[Z] = p1.z;
								acedTrans(pt,&wcs,&ucs,0,pt);
							}
							else
							{							
								p1=ptStart;
								pt[X] = p1.x;
								pt[Y] = p1.y;
								pt[Z] = p1.z;
								acedTrans(pt,&wcs,&ucs,0,pt);
							}
						}					
					}
				}
				else if (_tcscmp(szKword,_T("c")) == 0)
				{
					if (pPoly!=NULL)
					{
						int nCount =pPoly->numVerts();
						if (nCount>1)
						{
							pPoly->setClosed(true);
							break;
						}
					}
				}				
				continue;
			}
			else if (RTNORM != nRet)
			{
				break;
			}

			acedTrans(pt,&ucs,&wcs,0,ptWcs);
			AcGePoint3d p2=asPnt3d(ptWcs);			
			//处理多段线图形
			if (pPoly == NULL)
			{
				pPoly = new AcDbPolyline();
				pPoly->addVertexAt(0,AcGePoint2d(p1.x,p1.y));
				pPoly->addVertexAt(1,AcGePoint2d(p2.x,p2.y));
				//添加到当前数据库当前空间
				AddEntToCurSpace(pPoly);
				//添加对象到事务处理
				actrTransactionManager->addNewlyCreatedDBRObject(pPoly);
				//显示图形
				pPoly->draw();
			}
			else
			{
				int nCount = pPoly->numVerts();
				pPoly->addVertexAt(nCount,AcGePoint2d(p2.x,p2.y));
				//显示图形
				pPoly->draw();
			}		
			p1=p2;
		}

		if (pPoly == NULL)
		{
			//未创建多段线,终止事务
			actrTransactionManager->abortTransaction();
			return;
		}
		else
		{
			int nCount = pPoly->numVerts();
			if (nCount == 1)
			{
				//仅有一个点,终止事务
				actrTransactionManager->abortTransaction();
				return;
			}
		}
		//事务结束
		actrTransactionManager->endTransaction();

AddEntToCurSpace 添加实体对象到当前数据库当前空间

代码

	static Acad::ErrorStatus AddEntToCurSpace(AcDbEntity *pEnt)
	{
		AcDbObjectId objId=AcDbObjectId::kNull;
		return AddEntToCurSpace(pEnt,objId);
	}
	static Acad::ErrorStatus AddEntToCurSpace(AcDbEntity *pEnt,AcDbObjectId &objId)
	{
		if (pEnt == NULL)
		{
			return Acad::eNullEntityPointer;
		}
		AcDbBlockTableRecordPointer pBlkRcd(acdbHostApplicationServices()->workingDatabase()->currentSpaceId(),AcDb::kForWrite);
		if (Acad::eOk != pBlkRcd.openStatus())
		{
			return pBlkRcd.openStatus(); 
		}
		return pBlkRcd->appendAcDbEntity(objId,pEnt);
	}

结束语:使用事务管理器的时候,事务中的对象不能使用close()关闭对象了;

posted @ 2020-03-11 17:16  edata  阅读(1161)  评论(0编辑  收藏  举报