osg网格划分

创建结构体,用于储存三角形顶点和重心信息
struct TriangleSelf 
{
public:
	osg::Vec3f _p0;
	osg::Vec3f _p1;
	osg::Vec3f _p2;
	osg::Vec3f _centre;
	TriangleSelf(osg::Vec3f p0,osg::Vec3f p1,osg::Vec3f p2,osg::Vec3f centre)
	{
		_p0 = p0;
		_p1 = p1;
		_p2 = p2;
		_centre = centre;
	}
	TriangleSelf()
	{

	}
};

//创建结构体,用于储存格子编号和三角形
struct Classification
{
public:
	int _gridIndex;
	std::vector _triangleselfList;
	Classification(int &gridIndex)
	{
		_gridIndex=gridIndex;
	}
};

//创建nodevisitor,用于遍历节点,得到所有三角形的顶点
class MeshNodeVisitor : public osg::NodeVisitor
{
public:

	//构造函数	
	MeshNodeVisitor():osg::NodeVisitor(osg::NodeVisitor::TRAVERSE_ALL_CHILDREN)
	{
		m_TriPoints=new osg::Vec3Array;         //定义数组,用于保存索引号对应的顶点
	}

	//重载MatrixTransform节点的apply函数
	virtual void apply(osg::MatrixTransform &node)
	{
		traverse(node);
	}

	//重载group的apply()函数
	virtual void apply(osg::Group &group)
	{
		traverse(group);
	}

	//重载Geode的apply()函数,获取对应索引号的顶点数组m_TriPoints,三角形索引数组triangle和三角形中心数组centerPoint
	virtual void apply(osg::Geode &geode)
	{


		unsigned int drwnum = geode.getNumDrawables();
		for ( unsigned int i = 0; i < drwnum; i++ )
		{
			osg::ref_ptrgeometry= geode.getDrawable( i )->asGeometry();
			if ( !geometry )
				continue;

			for ( unsigned int n = 0; n < geometry->getNumPrimitiveSets(); ++n )
			{
				osg::PrimitiveSet* ps =geometry->getPrimitiveSet( n );				
				if ( !ps ) 
					continue;
				osg::ref_ptr va = dynamic_cast(geometry->getVertexArray());//获取顶点数组

				if((osg::PrimitiveSet::DrawElementsUIntPrimitiveType==ps->getType())&&(osg::PrimitiveSet::TRIANGLES==ps->getMode()))
				{
					osg::ref_ptrdeui = new osg::DrawElementsUInt;  //定义deui用于接收索引数组
					deui = dynamic_cast(ps);
					const unsigned indexNum = deui->getNumIndices(); //indexNum获取了索引的个数
					for (unsigned int m=0; mpush_back(va->at(deui->at(m)));//获取索引位置的顶点
					}
				}
				else
				{
					std::cout<<"模型绘制方式不是三角形!!"< getTriPoints()
	{
		return m_TriPoints.release();
	}

private:
	osg::ref_ptrdeui;
	osg::ref_ptr m_TriPoints;
};

//创建函数,用于绘制划分网格
osg::ref_ptrCreateGrids(const osg::Vec3f oriPoint,float lengthx,float lengthy,int &row, int &col, int&pace)

{	//申请一个叶子节点
	osg::ref_ptrpGroup =new osg::Group();	
	//申请变量用于表示各顶点所在的行、列以及顶点之间的间隔
	row=0;
	col=0;
	pace=0;

	//输出模型包围盒的大小,便于设置行列数。
	std::cout<<"包围盒x方向大小为:"<>col>>row>>pace;	

	for(int i=0;ivertexs = new osg::Vec3Array;
			//计算出任意一个四边形顶点的坐标值						
			vertexs->push_back(osg::Vec3f(x_ori+j*pace,y_ori-i*pace,z_ori));
			vertexs->push_back(osg::Vec3f(x_ori+(j+1)*pace,y_ori-i*pace,z_ori));
			vertexs->push_back(osg::Vec3f(x_ori+(j+1)*pace,y_ori-(i+1)*pace,z_ori));
			vertexs->push_back(osg::Vec3f(x_ori+j*pace,y_ori-(i+1)*pace,z_ori));

			//绘制格子
			osg::ref_ptr geode = new osg::Geode;
			osg::ref_ptrgeom=new osg::Geometry();
			geom->setVertexArray(vertexs);			
			geom->addPrimitiveSet(new osg::DrawArrays(osg::PrimitiveSet::QUADS,0,4));
			osg::ref_ptr colors2 = new osg::Vec4Array;
			colors2->push_back(osg::Vec4(0,0,0,0.1));
			geom->setColorArray(colors2);
			geom->setColorBinding(osg::Geometry::BIND_OVERALL);
			geode->addDrawable(geom.get());	
			geode->getOrCreateStateSet()->setMode(GL_BLEND,osg::StateAttribute::ON);
			int gridIndex = i*col+j;
			std::ostringstream oss;//创建一个流
			oss<setName(gridName);

			osg::ref_ptr geode2 = new osg::Geode;
			osg::ref_ptrgeom2=new osg::Geometry();
			geom2->setVertexArray(vertexs);
			osg::ref_ptr colors = new osg::Vec4Array;
			colors->push_back(osg::Vec4(1,1,1,1));
			geom2->setColorArray(colors);
			geom2->setColorBinding(osg::Geometry::BIND_OVERALL);
			geom2->addPrimitiveSet(new osg::DrawArrays(osg::PrimitiveSet::LINE_LOOP,0,4));
			geode2->addDrawable(geom2.get());	
			osg::LineWidth* lw = new osg::LineWidth;
			lw->setWidth(1);			
			geode2->getOrCreateStateSet()->setAttributeAndModes(lw,osg::StateAttribute::ON);

			pGroup->addChild(geode);
			pGroup->addChild(geode2);
		}		
	}	
	//函数返回geode
	return pGroup.release();
}

//创建函数,用于绘制包围盒,从而获取划分网格的参考位置以及参考大小
osg::ref_ptr createboundingbox(osg::Node *node,osg::Vec3f &oriPoint,float &lengthx,float &lengthy)
{
	osg::ref_ptr geode = new osg::Geode;
	osg::ComputeBoundsVisitor boundvisitor ;  
	node->accept(boundvisitor); 
	osg::BoundingBox bb = boundvisitor.getBoundingBox(); 

	//获取包围盒的长宽高和中点
	lengthx = bb.xMax()-bb.xMin();  
	lengthy = bb.yMax()-bb.yMin();  
	float lengthz = bb.zMax()-bb.zMin(); 
	osg::Vec3f pcenter=bb.center();

	//获取包围盒左后上方顶点
	oriPoint.set(bb.xMin(),bb.yMax(),bb.zMin());

	//绘制包围盒
	osg::ref_ptrdrawable = new osg::ShapeDrawable(new osg::Box(pcenter,lengthx,lengthy,lengthz)); 

	//设置包围盒样式
	osg::ref_ptrstateset = new osg::StateSet;  
	stateset= drawable->getOrCreateStateSet();
	osg::ref_ptrpolygon = new osg::PolygonMode(osg::PolygonMode::FRONT_AND_BACK,osg::PolygonMode::LINE);  
	stateset->setAttributeAndModes(polygon); 	
	//设置线宽
	osg::ref_ptrlinewidth = new osg::LineWidth(3.0);  
	stateset->setAttribute(linewidth);
	//设置包围盒颜色
	drawable->setColor(osg::Vec4(1.0,1.0,0.0,1.0));	

	geode->addDrawable(drawable);
	return geode.get();
}

//创建函数,用于判断一个点是否在矩形内部
bool isPointinPolygon(osg::Vec2f p,osg::ref_ptr pPoly)
{		
	float xMin=pPoly->at(3).x();//此处顶点的编号与后续本程序中网格顶点顺序有关
	float xMax=pPoly->at(1).x();
	float yMin=pPoly->at(3).y();
	float yMax=pPoly->at(1).y();
	if(xMin<=p.x()&&p.x() classFications):mViewer(viewer),lastselect(0)
	{
		mClassifications = classFications;
		pRoot = viewer->getSceneData()->asSwitch();
		pTriGeode = new osg::Geode();
	}//定义构造函数,有两个参数,mViewer和lastselect,分别初始化为viewer和0
	//在handle中实现的功能有双击鼠标左键调用pick函数
	virtual bool handle(const osgGA::GUIEventAdapter&ea,osgGA::GUIActionAdapter& aa)
	{
		osg::ref_ptrviewer=dynamic_cast(&aa);
		if(viewer=NULL)
			return false;

		switch(ea.getEventType())
		{
		case(osgGA::GUIEventAdapter::DOUBLECLICK):
			{
				if(ea.getButton()==1)
					Pick(ea.getX(),ea.getY());				
			}
			return true;
		}
		return false;
	}



private:
	//pick函数用于选择对象
	void Pick(float x,float y)
	{
		osg::ref_ptrnode=new osg::Node;
		osg::ref_ptrparent=new osg::Group;
		/*** 进行相交测试,几种测试方法见osgUtil::LineSegmentIntersector,结果保留在lsi里面***/
		osgUtil::LineSegmentIntersector* lsi=new osgUtil::LineSegmentIntersector(osgUtil::LineSegmentIntersector::WINDOW,x,y);
		osgUtil::IntersectionVisitor ivsec(lsi);
		mViewer->getCamera()->accept(ivsec);

		if (lsi->containsIntersections())
		{
			if (!(lsi->getFirstIntersection().nodePath.empty()))
			{
				osg::NodePath np= lsi->getFirstIntersection().nodePath;
				osg::ref_ptr pGeode = dynamic_cast(np[np.size()-1]);
				if(pGeode.valid())
				{
					std::string gridName = pGeode->getName();
					int num = atoi(gridName.c_str());
					std::vector triList = mClassifications[num]._triangleselfList;
					//更新三角形
					if(pTriGeode->getNumDrawables()>0)
					{
						pRoot->removeChild(pTriGeode);
						pTriGeode->removeDrawables(0,pTriGeode->getNumDrawables());
					}
					for (int j = 0 ;j pTriGeom = new osg::Geometry;
						osg::ref_ptr pTriVerts = new osg::Vec3Array;
						pTriVerts->push_back(osg::Vec3(tri._p0));
						pTriVerts->push_back(osg::Vec3(tri._p1));
						pTriVerts->push_back(osg::Vec3(tri._p2));
						pTriGeom->setVertexArray(pTriVerts);
						pTriGeom->addPrimitiveSet(new osg::DrawArrays(osg::PrimitiveSet::TRIANGLES,0,pTriVerts->size()));
						pTriGeode->addDrawable(pTriGeom);
					}
					pRoot->addChild(pTriGeode);
				}
			} 			
		}
		return;
	}

	osgViewer::Viewer *mViewer;
	osg::Node *lastselect;
	std::vector mClassifications;
	osg::ref_ptr pRoot;
	osg::ref_ptr pTriGeode;
};


int main ()
{
	osg::ref_ptrviewer=new osgViewer::Viewer;
	osg::ref_ptrroot=new osg::Switch;
	osg::ref_ptrmesh=osgDB::readNodeFile("mesh_2.osg");

	MeshNodeVisitor mnv;
	mesh->accept(mnv);

	//获取模型三角形信息
	osg::ref_ptrpVerts = mnv.getTriPoints();
	std::vector triangleVector;
	for (unsigned int i = 0;isize();i+=3)
	{
		osg::Vec3f p0 = pVerts->at(i);
		osg::Vec3f p1 = pVerts->at(i+1);
		osg::Vec3f p2 = pVerts->at(i+2);
		osg::Vec3f centre = (p0+p1+p2)/3.0;
		TriangleSelf _triangle(p0,p1,p2,centre);
		triangleVector.push_back(_triangle);
	}

	//调用函数createboundingbox()和CreateGrids(),用于绘制网格
	osg::Vec3f oriPoint;
	float lengthx,lengthy;
	int row,col,pace;
	osg::ref_ptr node = createboundingbox(mesh,oriPoint,lengthx,lengthy);
	osg::ref_ptr geode=CreateGrids(oriPoint,lengthx,lengthy,row,col,pace);

	//判断三角形和网格的关系,将三角形进行按网格划分
	std::vector classification;
	int gridNum = (col) * (row);
	for (int i = 0 ;isize();++k,n+=3)
	{
		for(int i=0;i pPoly = new osg::Vec2Array;
				pPoly->push_back(osg::Vec2f(oriPoint.x()+j*pace,oriPoint.y()-i*pace));
				pPoly->push_back(osg::Vec2f(oriPoint.x()+(j+1)*pace,oriPoint.y()-i*pace));
				pPoly->push_back(osg::Vec2f(oriPoint.x()+(j+1)*pace,oriPoint.y()-(i+1)*pace));
				pPoly->push_back(osg::Vec2f(oriPoint.x()+j*pace,oriPoint.y()-(i+1)*pace));
				if(isPointinPolygon(p,pPoly))
				{
					int gridIndex=i*col+j;
					classification[gridIndex]._triangleselfList.push_back(triangleVector[k]);						
					isNextTri = true;
					break;
				}
			}			
		}
	}
	
	
	//输出每个格子内有多少三角形
	for(int i=0;igriVertex=new osg::Vec3Array;
			griVertex->push_back(classification[m]._triangleselfList[n]._p0);
			griVertex->push_back(classification[m]._triangleselfList[n]._p1);
			griVertex->push_back(classification[m]._triangleselfList[n]._p2);

			osg::ref_ptrtriGeometry=new osg::Geometry;
			triGeometry->setVertexArray(griVertex);
			triGeometry->addPrimitiveSet(new osg::DrawArrays(osg::PrimitiveSet::TRIANGLES,0,3));

			osg::ref_ptrtriGeode=new osg::Geode;
			triGeode->addDrawable(triGeometry);			
			triGroup->addChild(triGeode.get());
		}
		std::ostringstream oss;//创建一个流
		oss<(triGroup);//在使用writeNodeFile时,节点始终是Node指针,故需要将其转换成Node
		osgDB::writeNodeFile(*triNode,fileName);
	}
	
	
	root->addChild(geode.get());
	//root->addChild(mesh.get(),true);
	viewer->setSceneData(root);
	viewer->addEventHandler(new PickHandler(viewer,classification));
	//viewer->setUpViewInWindow(500,500,800,600);
	return viewer->run();
}

你可能感兴趣的:(OpenGL)