青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆-38  評論-23  文章-0  trackbacks-0

      關(guān)于碰撞檢測,始終是物理系統(tǒng)在圖形學(xué)運(yùn)用上的一個比較復(fù)雜的問題.碰撞檢測做的好不好.完全決定一個場景漫游的逼真性.
     這幾天,在坐城市汽車仿真處理上,對于驅(qū)動汽車運(yùn)動時候,對于汽車的碰撞檢測問題困擾了我相當(dāng)?shù)木?始終沒能做到很好.這當(dāng)中我現(xiàn)在使用的汽車包圍體對場景進(jìn)行求交測試時候,用到了OSG中PolytopeIntersector.
     對于一個Polytope 應(yīng)當(dāng)是多個平面組成的一個空間區(qū)域,對于OSG求交器當(dāng)中,這個多面體各個平面的正面必須都是屬于這個區(qū)域內(nèi)的? 怎么解釋呢.平面的法線應(yīng)當(dāng)是指向該空間區(qū)域內(nèi)部.比如說一個四面體.其內(nèi)部的四個面應(yīng)當(dāng)是正面(法線朝向的方向那個面.) 這是多面體求交器使用的一個關(guān)鍵.(這在之后的代碼解讀當(dāng)中會一并解釋).

     對于OSG場景求交測試,是必然要用到訪問器的(Vistor.這個應(yīng)當(dāng)在在另辟一篇文章才能詳述清楚,它的使用原理,因此這里我們暫時先用著.) 對于求交器使用到的訪問器(Vistor)應(yīng)當(dāng)是交集訪問器(IntersectionVistor).

    因此,我們在定義上則應(yīng)當(dāng)是如下:

    /** 創(chuàng)建一個多面體求交器*/
    osgUtil::PolytopeIntersector
* pI =new osgUtil::PolytopeIntersector(poly);
    
/** 構(gòu)造一個交集訪問器*/
    osgUtil::IntersectionVisitor iv(pI);
    對于求交集 則應(yīng)當(dāng)對于場景根節(jié)點(diǎn)做請求訪問的操作..這當(dāng)中可能需要避開自身節(jié)點(diǎn)等一些不必要的節(jié)點(diǎn)等.
    /** 設(shè)置避開自身節(jié)點(diǎn)*/
    _model
->setNodeMask(0x0);
    
/** 根節(jié)點(diǎn)請求訪問操作*/
    root
->accept(iv);
    
/** 恢復(fù)自身節(jié)點(diǎn)的NodeMask*/
    _model
->setNodeMask(0xffffffff);

     對于setNodeMask()避開節(jié)點(diǎn)等.我想應(yīng)當(dāng)在Vistor中在詳述..
     再對于訪問操作之后,我們就可以獲得所返回的交集了.
    
    if(pI->containsIntersections())
    
{
        typedef osgUtil::PolytopeIntersector::Intersections inters;
        
for(inters::iterator it=pI->getIntersections().begin();\
            it
!=pI->getIntersections().end();it++)
            
/** ……*/
    }

     固然,這些只是相對于簡單的操作.而我們是想要深入到了解在root->accept(iv)之后到底做了什么事情?它到底如何求得了我們想要的數(shù)據(jù)? 那現(xiàn)在開始我們的代碼解讀之旅……當(dāng)然這其中,我想有必要略去一些關(guān)系到Vistor的內(nèi)容.因?yàn)檫@些詳述起來,不是簡短的能夠說的清楚...

     現(xiàn)在我們定位到: osgUtil/IntersectionVisitor.cpp 第226行:
     

apply(osg::Geode& geode)

 

    因?yàn)間eode是葉子節(jié)點(diǎn),最后肯定都會請求到它,并訪問..其中的代碼我們將能夠非常直觀的看出它將要干嘛?
對于geode下的所有可繪制圖元進(jìn)行求交.因此我們現(xiàn)在將轉(zhuǎn)到 intersect函數(shù)

    定位到: include/osgUtil/IntersectionVisitor.h 第245行:
   

intersect


    關(guān)于_intersectorStack 是個求交器的集合,我們在構(gòu)造的時候?qū)olytopeIntersector傳入后將會被加入到這個集合當(dāng)中..因此 這將會回到PolytopeIntersector中的intersect函數(shù)..因此,我們又得重新打開polytope那個文件咯..
    定位到 osgUtil/PolyIntersector.cpp 第547行..
   

 

PolytopeIntersector::intersect


     我們可以看到再用 PolytopePrimitiveIntersector 構(gòu)造了一個func 后(并設(shè)置多面體,和參考平面) 對于drawable進(jìn)行訪問操作?似乎又回到vistor...? 其實(shí)這個只是類似的操作,但還算不上vistor..暫時當(dāng)作類似的對待吧..雖然Vistor模式在OSG中的運(yùn)用非常的多..而且?guī)缀跆幪幎紩玫?.這個時候我們將要進(jìn)入一個關(guān)鍵時刻,因?yàn)槲覀冎?在osg中drawable里頭已經(jīng)是最后的頂點(diǎn)等所有數(shù)據(jù)存放的地方.drawable其實(shí)只是個抽象類.這里我只會簡單的通過它的一個特例:Geometry 來講述這一段內(nèi)容..
所以現(xiàn)在  我們將定位在osg/Geometry.cpp 第2199行:
     

void Geometry::accept(PrimitiveFunctor& functor) const

      對于這個操作,我們暫時只看不存在索引數(shù)據(jù)的..因?yàn)橄鄬τ趤碇v原理總是一樣的.后面的只是多了一些步驟將頂點(diǎn)數(shù)據(jù)取出..好了.我們回到正題.
       functor.setVecterArray() 很直觀的明白,將頂點(diǎn)數(shù)據(jù)存到fuctor里.以便于在之后functor操作.
      
      其后最主要的還是在于對于drawable里的每個primitiveset 進(jìn)行接受fuctor訪問操作 (*itr)->accept(functor);
我們知道.primitiveset里頭擁有的數(shù)據(jù)是頂點(diǎn)最后繪制的規(guī)則. 相當(dāng)于我們在OPENGL當(dāng)中使用glBegin() glEnd()一樣指定最后基礎(chǔ)圖元生成的規(guī)則.而我們所要求交集的目的在于獲得跟這些基礎(chǔ)圖元的交集.因此.我們有必要繼續(xù)往下深究.我們還沒有嗅到最終結(jié)果,還不能夠放棄. 好了 繼續(xù)..PrimitiveSet又是一個虛類.因此,我們有必要挑個實(shí)體類來深究.就選DrawArray吧. DrawArray指定一個MODE,頂點(diǎn)的起始位置,以及參與繪制的頂點(diǎn)的總數(shù)目..
MODE 就相當(dāng)于 GL_LINES GL_TRIANGLES 等等.我們再次回到代碼來說吧.
       這次我們將定位在: osg/PrimitiveSet.cpp 第43行:
       很簡單...
DrawArrays::accept
        所以最終的結(jié)果 都將回到fuctor里頭進(jìn)行交集運(yùn)算的處理..._mode _first _count 將擁有的規(guī)則送往fuctor..

       在追究了這么多之后,我們又需要回到functor里頭.這個functor 是什么呢? 還記得我之前說的使用PolytopePrimitiveIntersector 構(gòu)造了一個func對不? 所有的關(guān)鍵將在那里揭開....最后的結(jié)果總還是深藏于原來的最初的起點(diǎn)位置..不過我想還真不枉繞了一圈...
      在我們回到func 之前我們還需要深究下functor.drawArrays() 這個函數(shù)到底做了什么? 因?yàn)樵赑olytopePrimitiveIntersector當(dāng)初我們并未發(fā)現(xiàn)有這個函數(shù).PolytopePrimitiveIntersector這個類是在PolytopeIntersector.cpp文件當(dāng)中定義的.它只有一大堆的operator()操作...因此我們需要回到構(gòu)造它的那個functor()里頭..
現(xiàn)在我將定位到 include/osg/TemplatePrimitiveFunctor.h  第90行..
     
drawArrays(GLenum mode,GLint first,GLsizei count)

        對于此,我們暫時只觀看最簡單的GL_TRIANGLES ,對于三角形的每三個點(diǎn)將會繪制一個三角形.因此每次只取三個頂點(diǎn),將它傳遞給當(dāng)前構(gòu)造的func0>operator()處理.這就是為什么 func里頭全部是都是operator()操作了..
       我們弄明白這些之后,馬上回到PolytopePrimitiveIntersector 最后的結(jié)果.令人期待啊...
       PolytopePrimitiveIntersector中的operator()支持很多種類型,.參數(shù)的不同,一個點(diǎn)(points)(兩個點(diǎn))lines,三個點(diǎn)(Triangles),四個點(diǎn)(quads)

       最后定位在三角形的處理上: osgUtil/PolytopeIntersector.cpp 第208行.

       這段代碼相當(dāng)?shù)拈L,但是看起來非常的好理解.這里我也將解釋為什么對于多面體在定義的時候法線很重要了?  我想我有必要將這部分代碼全部解讀清楚..這部分是關(guān)鍵.
void operator()

現(xiàn)在將做最后的代碼解讀工作 selector_mask 當(dāng)前操作平面編號的標(biāo)記 inside_mask 標(biāo)記三角形在哪些平面的正面?即所說在區(qū)域內(nèi)..對于所有平面,將進(jìn)行如下操作:
1.  d1 d2 d3 分別求得 ax+by+cz+d < = > 0 
     [ax+by+cz >0 表示點(diǎn)在正面這邊,=0 表示點(diǎn)平面上,<0則表示在背面這邊] 
     若三個點(diǎn)都在某個平面的背面..那說明這個三角形肯定在這個多面體的區(qū)域外.則結(jié)束..
     若三個點(diǎn)都在某個平面的正面,則做標(biāo)記并繼續(xù)其他平面.
2.  若不是以上兩種情況,那分別判斷v1v2 v1v3 v2v3這三條線段的與平面的交點(diǎn).并加入至候選頂點(diǎn)列表當(dāng)中.

在對所有平面都進(jìn)行操作之后,需要判斷幾種情況我們可以考慮?
第一.三角形剛好在多面體內(nèi)部.
第二.可能這些交點(diǎn)落在其他平面的背面了.
第三 可能三條邊與平面是存在交點(diǎn).但是多面體的組成的閉合區(qū)域卻剛好穿過三角形內(nèi)部.這個時候必須對平面的交線與三角形求交點(diǎn)..

所以這三個部分完全概括了上面的代碼?是的.我想這個部分并不需要我講的有多么詳細(xì)了.很容易理解的.

其后,我還想深究下最后這個交集會存放到哪里去了?我們最終該如何使用獲得交集才能夠更好被我們所利用?

    addIntersection(_index, _candidates);

    對于每處理一個三角形 _index 都會在開頭部分自增..因此 對于Intersections中的每一個交集的點(diǎn)都針對于同一個三角形..(對于別的同理可得?) 也就是說_index表示在primitiveSet當(dāng)中.這個三角形是第幾個三角形.(三角形序號)
    最后,我們再次回到我們最開始進(jìn)入這么大段篇幅討論的起始位置吧?還記得否?我們第二個intersect()函數(shù)..就是PolytopeIntersector類中的..因?yàn)槲覀冏詈蟮慕Y(jié)果總會回歸到我們需要的地方.所以我們現(xiàn)在得回到那里去取得我們最終獲得的數(shù)據(jù)/.
   
結(jié)果如何?

      對于從func中獲得的交集.我們將需要將它變成我們所需要的數(shù)據(jù).我將一一解釋最終我們得到的每個數(shù)據(jù)的含義:
     hit.distance // 表示從當(dāng)前這個交集的所有頂點(diǎn)的中心點(diǎn)到參考平面的距離.
     hit.primitiveIndex //表示之前我們說的這個圖元在PrimitiveSet中的序號.
     hit.nodepath  //表示這個從根結(jié)點(diǎn)到當(dāng)前這個geode的路徑..因?yàn)槲覀冎涝趘istor中我們有pushNodepath() popNodePath()來保存這個路徑操作..所以這個路徑是從vistor中獲得的.
     hit.drawable //當(dāng)然是我們保存著當(dāng)前這個交集是對于哪個drawable.
     hit.matrix   //表示當(dāng)前這個drawable應(yīng)當(dāng)在世界坐標(biāo)系的變換矩陣.我們可以使用point*matrix 來得到獲得點(diǎn)在世界坐標(biāo)系下的位置..
     hit.localIntersecotPoint  //表示所有交點(diǎn)的中心點(diǎn).
     hit.intersectorPoint //所有交點(diǎn)的一個數(shù)組..目前最多的頂點(diǎn)個數(shù)應(yīng)該是6..  enum { MaxNumIntesectionPoints=6 };
     hit.numintersectorPoint // 所有頂點(diǎn)的個數(shù)..

我想這個解讀過程到此應(yīng)當(dāng)結(jié)束了...繼續(xù)學(xué)習(xí)ING.....


posted on 2009-08-02 15:03 米游 閱讀(7577) 評論(1)  編輯 收藏 引用 所屬分類: OpenGL/OSG

評論:
# re: OSG 碰撞檢測之多面體求交器代碼解讀(PloytopeIntersector) 2018-05-03 14:49 | 盧江
你好,能不能分享一下你寫的這個碰撞檢測,多面體求交的源碼呀?我最近在寫這個碰撞檢測的代碼上碰到好多問題,希望能參考一下你的代碼,不勝感激!(我的郵箱:313741269@qq.com)  回復(fù)  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            久久蜜桃精品| 欧美午夜精品一区| 嫩草影视亚洲| 国产日韩在线一区| 中文日韩在线视频| 亚洲国产成人一区| 久久久久久高潮国产精品视| 国产精品v欧美精品∨日韩| 99精品欧美一区| 亚洲欧洲日产国产综合网| 久久综合久久88| 亚洲高清资源| 欧美成人有码| 欧美.www| 亚洲激情电影中文字幕| 蜜桃久久精品一区二区| 久久精品国产欧美亚洲人人爽| 国产日产欧美精品| 久久国产黑丝| 久久久www成人免费毛片麻豆| 国产一区亚洲| 美女露胸一区二区三区| 久久夜色精品国产亚洲aⅴ| 亚洲电影视频在线| 亚洲成人在线视频播放| 欧美黄色aa电影| 99re国产精品| 亚洲性感美女99在线| 国产乱码精品一区二区三区不卡| 久久国产精品一区二区| 久久经典综合| 亚洲欧洲一区二区在线观看| 亚洲日本理论电影| 国产精品国产三级国产普通话三级 | 妖精成人www高清在线观看| 亚洲国产91精品在线观看| 欧美国产视频日韩| 欧美日韩在线电影| 国产精品劲爆视频| 久久精品一区二区三区中文字幕| 久久精品国产精品亚洲综合| 91久久精品一区二区三区| 91久久久在线| 国产精品多人| 久久人体大胆视频| 欧美精品久久久久久久免费观看| 亚洲一区视频| 久久三级视频| 亚洲制服少妇| 蜜桃精品一区二区三区| 亚洲性感美女99在线| 亚洲欧美日韩综合| 亚洲国产精品成人| 亚洲夜间福利| 亚洲激情一区二区| 午夜精品久久久久久久99热浪潮| 亚洲二区在线观看| 亚洲一区制服诱惑| 亚洲人成亚洲人成在线观看图片 | 免费在线看一区| 欧美日韩在线播放一区二区| 老司机aⅴ在线精品导航| 欧美三级视频在线| 欧美第一黄网免费网站| 国产精品主播| 99av国产精品欲麻豆| 一区在线观看| 午夜精品免费视频| 一区二区久久| 欧美高清一区二区| 美女久久网站| 好看的亚洲午夜视频在线| 99在线热播精品免费99热| 亚洲欧洲日本在线| 久久夜色精品国产| 久久欧美肥婆一二区| 国产精品一区二区久久精品| 亚洲精品视频免费在线观看| 在线免费观看日本欧美| 欧美一区二区三区在线观看| 亚洲欧美高清| 欧美性猛交xxxx免费看久久久 | 午夜亚洲视频| 亚洲免费在线播放| 欧美日韩免费区域视频在线观看| 欧美成人四级电影| 一区二区三区在线免费视频| 亚洲欧美成人| 久久国产日本精品| 国产日韩一区欧美| 午夜在线观看欧美| 久久国产精品久久久| 国产精品久久久久久久久久免费| 日韩一级黄色大片| 中文欧美字幕免费| 国产精品国产精品国产专区不蜜| 亚洲视频免费观看| 亚洲伊人第一页| 亚洲品质自拍| 美女主播精品视频一二三四| 久久午夜色播影院免费高清| 久久久精品国产99久久精品芒果| 欧美日一区二区三区在线观看国产免| 一本色道久久综合亚洲精品高清| 一区二区三区四区五区视频| 国产一区二区三区高清播放| 久久久之久亚州精品露出| 麻豆精品视频在线观看| 在线天堂一区av电影| 久久成人免费网| 亚洲另类一区二区| 欧美中文字幕在线| 亚洲美女一区| 久久成人18免费观看| 亚洲一区二区三区精品在线观看 | 欧美韩国一区| 国产欧美一区二区精品仙草咪| 亚洲国产欧美一区二区三区丁香婷| 国产欧美日本一区二区三区| 日韩一级欧洲| 99精品视频免费观看| 久久久青草婷婷精品综合日韩 | 久久久久久9999| 欧美视频一二三区| 久久亚洲综合色| 国产一区二区三区直播精品电影 | 欧美日韩亚洲高清一区二区| 嫩草国产精品入口| 国产在线拍偷自揄拍精品| 一区二区不卡在线视频 午夜欧美不卡在 | 免费美女久久99| 欧美二区不卡| 韩国av一区二区三区四区| 先锋影音久久| 久久久国产精品亚洲一区 | 欧美激情1区2区| 亚洲福利av| 六月婷婷一区| 亚洲高清在线视频| 日韩午夜在线播放| 国产精品成人免费| 午夜免费在线观看精品视频| 久久精品国产免费看久久精品| 国产日韩在线一区| 久久看片网站| 亚洲精品国产品国语在线app| 99视频超级精品| 国产精品免费网站| 欧美在线国产| 免费观看30秒视频久久| 亚洲国产日韩欧美| 欧美片第一页| 亚洲欧美日韩成人高清在线一区| 久久精品国内一区二区三区| 亚洲成在人线av| 欧美区一区二区三区| 亚洲午夜免费视频| 久久久高清一区二区三区| 亚洲电影免费观看高清完整版在线 | 伊人狠狠色丁香综合尤物| 久久综合狠狠综合久久激情| 亚洲高清视频的网址| 亚洲一区欧美| 国产亚洲一本大道中文在线| 久久精品人人做人人爽| 亚洲国产精品久久久久秋霞不卡| 中文精品视频| 激情视频一区二区| 欧美三级精品| 久久国产精品久久w女人spa| 亚洲黄色小视频| 欧美一区二区三区喷汁尤物| 亚洲激情成人在线| 国产精品中文字幕在线观看| 久久精品一级爱片| 亚洲美女在线观看| 欧美 日韩 国产一区二区在线视频| 日韩视频一区二区在线观看 | 国产亚洲激情在线| 欧美日本韩国一区二区三区| 久久精品在线播放| 一区二区av在线| 亚洲国产成人一区| 久久久久一区二区| 午夜精品网站| 99视频一区二区| 亚洲国产精品传媒在线观看| 国产精品外国| 欧美日韩a区| 美女在线一区二区| 欧美有码视频| 亚洲男人av电影| 亚洲免费观看在线视频| 欧美电影资源| 欧美成人精品不卡视频在线观看 | 亚洲综合日韩中文字幕v在线| 亚洲国产小视频在线观看| 亚洲国产小视频| 国产综合久久| 国产一区二区日韩精品|