您好,登錄后才能下訂單哦!
這篇文章主要介紹“OpenSceneGraph如何導出三角形數據”,在日常操作中,相信很多人在OpenSceneGraph如何導出三角形數據問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”OpenSceneGraph如何導出三角形數據”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!
在OpenSceneGraph開發中,為了方便會經常使用到一些不是三角形片的數據,比如四邊形等數據。例如畫一個管子用四邊形帶比用三角形片好計算得多。比如現在我們要畫一個由兩個平面組成的面,我可以這樣做:
osg::Geode* geode=new osg::Geode; osg::Geometry* polyGeom = new osg::Geometry; osg::Vec3 myCoords[]= { osg::Vec3(0,1,0), osg::Vec3(0,0,0), osg::Vec3(1,1,0), osg::Vec3(1,0,0), osg::Vec3(2,1,0), osg::Vec3(2,0,0) }; int numCoords = sizeof(myCoords)/sizeof(osg::Vec3); osg::Vec3Array* vertices = new osg::Vec3Array(numCoords,myCoords); polyGeom->setVertexArray(vertices); polyGeom->addPrimitiveSet(new osg::DrawArrays(osg::PrimitiveSet::QUAD_STRIP,0,numCoords)); geode->addDrawable(polyGeom);
這樣就用6個點,用OpenGL提供的QUAD_STRIP方式畫出了兩個平面。 但是如果要把這個平面用于碰撞檢測等技術,那么就需要把這六個點所表示的四邊形帶轉換成三角形片才行。這些三角形定點如下:
0 1 0 0 0 0 1 1 0 0 0 0 1 0 0 1 1 0 1 1 0 1 0 0 2 1 0 1 0 0 2 0 0 2 1 0
可以看出兩個平面由4個三角形組成,而且都是逆時針排列(朝向一致)。 以前我自己做過轉換,但是感覺很麻煩。OpenSceneGraph的Example osggeometry中提供了一個printTriangles函數,它可以打印出一個drawable所有的三角形片,不管最初的數據結構如何:
struct NormalPrint { void operator() (const osg::Vec3& v1,const osg::Vec3& v2,const osg::Vec3& v3, bool) const { osg::Vec3 normal = (v2-v1)^(v3-v2); normal.normalize(); std::cout << "t("<<<") ("<<<") ("<<<") "<<") normal ("<<<")"< } }; // decompose Drawable primtives into triangles, print out these triangles and computed normals. void printTriangles(const std::string& name, osg::Drawable& drawable) { std::cout<< osg::TriangleFunctor tf; drawable.accept(tf); std::cout< }
核心的思想就是利用osg::TriangleFunctor這個模版。這個模版會讓你重載()運算符,然后讓Drawable去visit它。在這個過程中,所有原始的數據(不管是三角形片的,還是四邊形的)都轉換成了三角形片數據。
那么如何把三角形數據導出哪?只需要修改一下借助這個思路,將NormalPrint修改成我們需要的就對了。
struct GetVertex { void operator() (const osg::Vec3& v1,const osg::Vec3& v2,const osg::Vec3& v3, bool) const { vertexList->push_back(v1); vertexList->push_back(v2); vertexList->push_back(v3); } osg::Vec3Array* vertexList; }; void getTriangles(osg::Drawable& drawable) { osg::TriangleFunctor tf; tf.vertexList=new osg::Vec3Array; drawable.accept(tf); for(osg::Vec3Array::iterator itr=tf.vertexList->begin(); itr!=tf.vertexList->end(); itr++) { osg::Vec3 vertex=*itr; std::cout<< } std::cout< }
以下是完整的示例文件:
// PrimitiveSet.cpp : 定義控制臺應用程序的入口點。 // #include "stdafx.h" #include struct GetVertex { void operator() (const osg::Vec3& v1,const osg::Vec3& v2,const osg::Vec3& v3, bool) const { vertexList->push_back(v1); vertexList->push_back(v2); vertexList->push_back(v3); } osg::Vec3Array* vertexList; }; void getTriangles(osg::Drawable& drawable) { osg::TriangleFunctor tf; tf.vertexList=new osg::Vec3Array; drawable.accept(tf); for(osg::Vec3Array::iterator itr=tf.vertexList->begin(); itr!=tf.vertexList->end(); itr++) { osg::Vec3 vertex=*itr; std::cout<< } std::cout< } osg::Node* createGeode() { osg::Geode* geode=new osg::Geode; osg::Geometry* polyGeom = new osg::Geometry; osg::Vec3 myCoords[]= { osg::Vec3(0,1,0), osg::Vec3(0,0,0), osg::Vec3(1,1,0), osg::Vec3(1,0,0), osg::Vec3(2,1,0), osg::Vec3(2,0,0) }; int numCoords = sizeof(myCoords)/sizeof(osg::Vec3); osg::Vec3Array* vertices = new osg::Vec3Array(numCoords,myCoords); polyGeom->setVertexArray(vertices); polyGeom->addPrimitiveSet(new osg::DrawArrays(osg::PrimitiveSet::QUAD_STRIP,0,numCoords)); geode->addDrawable(polyGeom); getTriangles(*polyGeom); return geode; } int _tmain(int argc, _TCHAR* argv[]) { //Set up viewer osgViewer::Viewer viewer; osg::ref_ptr traits=new osg::GraphicsContext::Traits; traits->x=200; traits->y=200; traits->width=800; traits->height=600; traits->windowDecoration=true; traits->doubleBuffer=true; traits->sharedContext=0; osg::ref_ptr gc=osg::GraphicsContext::createGraphicsContext(traits.get()); osg::ref_ptr camera=new osg::Camera; //osg::Camera camera=new osg::Camera; camera->setGraphicsContext(gc.get()); camera->setViewport(new osg::Viewport(0,0,traits->width,traits->height)); camera->setDrawBuffer(GL_BACK); camera->setReadBuffer(GL_BACK); osgGA::TrackballManipulator* tm=new osgGA::TrackballManipulator; viewer.setCameraManipulator(tm); viewer.addSlave(camera.get()); //Set up root node osg::ref_ptr root=new osg::Group; root->addChild(createGeode()); //Start show! viewer.setSceneData(root.get()); viewer.realize(); while(!viewer.done()) { viewer.frame(); } }
到此,關于“OpenSceneGraph如何導出三角形數據”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。