• <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>
            xiaoguozi's Blog
            Pay it forword - 我并不覺的自豪,我所嘗試的事情都失敗了······習(xí)慣原本生活的人不容易改變,就算現(xiàn)狀很糟,他們也很難改變,在過程中,他們還是放棄了······他們一放棄,大家就都是輸家······讓愛傳出去,很困難,也無法預(yù)料,人們需要更細心的觀察別人,要隨時注意才能保護別人,因為他們未必知道自己要什么·····

            1. OpenGL 渲染管線

            OpenGL渲染管線分為兩大部分,模型觀測變換(ModelView Transformation)投影變換(Projection Transformation)。做個比喻,計算機圖形開發(fā)就像我們照相一樣,目的就是把真實的場景在一張照相紙上表現(xiàn)出來。那么觀測變換的過程就像是我們擺設(shè)相機的位置,選擇好要照的物體,擺好物體的造型。而投影變換就像相機把真實的三維場景顯示在相紙上一樣。下面就分別詳細的講一下這兩個過程。

            1.1模型觀測變換

            讓我們先來弄清楚OpenGL中的渲染管線。管線是一個抽象的概念,之所以稱之為管線是因為顯卡在處理數(shù)據(jù)的時候是按照一個固定的順序來的,而且嚴格按照這個順序。就像水從一根管子的一端流到另一端,這個順序是不能打破的。先來看看下面的圖1:

            1                     圖1 OPENGL渲染管線                                 

            圖中顯示了OpenGL圖形管線的主要部分,也是我們在進行圖形編程的時候常常要用到的部分。一個頂點數(shù)據(jù)從圖的左上角(MC)進入管線,最后從圖的右下角(DC)輸出MC是Model Coordinate的簡寫,表示模型坐標(biāo)DCDevice Coordinate的簡寫,表示設(shè)備坐標(biāo)。 當(dāng)然DC有很多了,什么顯示器,打印機等等。這里DC我們就理解成常說的屏幕坐標(biāo)好了。MC當(dāng)然就是3D坐標(biāo)了(注意:我說的3D坐標(biāo),而不是世界坐 標(biāo)),這個3D坐標(biāo)就是模型坐標(biāo),也說成本地坐標(biāo)(相對于世界坐標(biāo))。MC要經(jīng)過模型變換(Modeling Transformation)才變換到世界坐標(biāo),圖2:

            2圖2 世界坐標(biāo)系和模型坐標(biāo)系

            變換到世界坐標(biāo)WC(World Coordinate)說簡單點就是如何用世界坐標(biāo)系來表示本地坐標(biāo)系中的坐標(biāo)。為了講得更清楚一些,這里舉個2D的例子。如圖3:

            3 圖3 世界坐標(biāo)系和模型坐標(biāo)系的計算

            圖中紅色坐標(biāo)系是世界坐標(biāo)系WC,綠色的是模型坐標(biāo)系MC。現(xiàn)在有一個頂點,在模型坐標(biāo)系中的坐標(biāo)為(1,1),現(xiàn)在要把這個模型坐標(biāo)轉(zhuǎn)換到世界坐標(biāo)中來表示。從圖中可以看出,點(1,1)在世界坐標(biāo)系中的坐標(biāo)為(3,4),現(xiàn)在我們來通過計算得到我們希望的結(jié)果。首先我們要把模型坐標(biāo)系MC在世界坐標(biāo)系中表示出來,使用齊次坐標(biāo)(Homogeneous Coordinate )可以表示為矩陣(注意,本教程中使用的矩陣都是以列向量組成):gif.latex 其中,矩陣的第一列為MC中x軸在WC中的向量表示第二列為MC中y軸WC中的向量表示第三列為MC中的原點在WC中的坐標(biāo)。對齊次坐標(biāo)系不了解的同學(xué),請先學(xué)習(xí)游戲數(shù)學(xué)方面的知識。有了這個模型變換矩陣后,用這個矩陣乘以在MC中表示的坐標(biāo)就可以得到該坐標(biāo)在世界坐標(biāo)系中的坐標(biāo)。所以該矩陣和MC中的坐標(biāo)(1,1)相乘有:

            gif.latex2 這也正是我們需要的結(jié)果。現(xiàn)在讓我們把相機坐標(biāo)也加進去,相機坐標(biāo)也稱為觀測坐標(biāo)(View Coordinate),如圖4和圖5。

            4 圖4 ModelView變換的三個坐標(biāo)系

            5 圖5 ModelView變換計算

            來看看MC坐標(biāo)中的點(1,1)如何在相機坐標(biāo)中表示。從圖5中可以直接看出MC中的點(1,1)在相機坐標(biāo)系VC中為(-2,-2)。和上面同樣的道理,我們可以寫出相機坐標(biāo)系VC在世界標(biāo)系WC中可以表示為:

            gif.latex3 那么世界坐標(biāo)系中的點轉(zhuǎn)換為相機坐標(biāo)系中的點我們就需求VC的逆矩陣:

            gif.latex4那么世界坐標(biāo)系WC中的點(3,4)在相機坐標(biāo)系VC中坐標(biāo)為:

            gif.latex5 上面的變換過程,就是可以把模型坐標(biāo)變換為相機坐標(biāo)。在OpenGL中,當(dāng)我們申明頂點的時候,有時候說的是世界坐標(biāo),這是因為初始化的時候世界坐標(biāo)系、模型坐標(biāo)系和相機坐標(biāo)系是一樣的,重合在一起的。所以O(shè)penGL中提供了模型觀測變換,它是把模型坐標(biāo)系直接轉(zhuǎn)換為相機坐標(biāo)系,如圖4。現(xiàn)在我們已經(jīng)計算得到了VC-1和MC,如果把VC-1和 MC相乘,就可以得到模型坐標(biāo)在相機坐標(biāo)中的表示。為了得到模型坐標(biāo)系中的坐標(biāo)在相機坐標(biāo)系中的表示,這就是OpenGL中的ModelView變換矩 陣。這也是ModelView變換的名字的由來,它是通過了上面兩個步驟得到的。那么這里,ModelView變換矩陣M為:

            gif.latex6 現(xiàn)在只要用上面的模型觀測矩陣M乘以模型坐標(biāo)系MC中的坐標(biāo)就可以得到相機坐標(biāo)系中的坐標(biāo)了。模型觀測變換的關(guān)鍵就是要得到相機坐標(biāo)系中的坐標(biāo),因為光照等計算都是在這個這個坐標(biāo)系中完成的。下面我們實際OpenGL程序中檢查一下。在程序中,為了計算方便,我們使用圖6中的模型。

            6 圖6 ModelView變換計算模型

            根據(jù)圖中的數(shù)據(jù),我們分別可以寫出對應(yīng)MC和VC-1,從而求得觀測變換矩陣M

            gif.latex7 現(xiàn)在程序中用glGetFloatv()這個函數(shù)來獲得當(dāng)前矩陣數(shù)據(jù)來檢查一下。

            1. float m[16] = {0}; //用來保存當(dāng)前矩陣數(shù)據(jù)  
            2. glMatrixMode(GL_MODELVIEW);  
            3. glLoadIdentity();  
            4. glGetFloatv(GL_MODELVIEW_MATRIX, m);   
            5. //相機設(shè)置,View 變換  
            6. gluLookAt(0.0, 0.0, 5.0,  
            7. 0.0, 0.0, 0.0,  
            8. 0.0, 1.0, 0.0);  
            9. glGetFloatv(GL_MODELVIEW_MATRIX, m);   
            10. //投影設(shè)置  
            11. glMatrixMode(GL_PROJECTION);  
            12. glLoadIdentity();  
            13. glOrtho(-10,10,-10,10,-10,10);  
            14. glMatrixMode(GL_MODELVIEW);   
            15. //Modeling變換  
            16. glTranslatef(0, 0, -3);  
            17. glGetFloatv(GL_MODELVIEW_MATRIX, m);  
            18. glBegin(GL_POINTS);  
            19. glVertex3f(1,1,0);  
            20. glEnd();  

            如果在上面程序段中最后一個glGetFloatv(GL_MODELVIEW_MATRIX, m)處設(shè)定斷點的話,就可以看到圖7所顯示的數(shù)據(jù)。

            7 圖7 ModelView變換矩陣數(shù)據(jù)

            到這里,整個ModelView變換就完成了。通過ModelView變換后得到是相機坐標(biāo)系內(nèi)的坐標(biāo)。在這個坐標(biāo)系內(nèi)典型的計算就是法線了。現(xiàn)在再來看看后面一個階段。

            1.2投影變換

            先還是復(fù)習(xí)一下OpenGL的渲染管線。圖1中可以看到,在投影變換(Projection Transformation)中也分為兩個部分,第一個部分是將上個階段得到的坐標(biāo)轉(zhuǎn)換為平面坐標(biāo),第二個部分是將轉(zhuǎn)換后的平面坐標(biāo)在進行歸一化并進行剪裁。一般地,將三維坐標(biāo)轉(zhuǎn)換為平面坐標(biāo)有兩種投影方式:正交投影(Orthogonal Projection)和透視投影(Perspective Projection)

            1.2.1 正交投影

            正交投影很簡單,如圖8,對于三維空間中的坐標(biāo)點和一個二維平面,要在對應(yīng)的平面上投影,只需將非該平面上的點的坐標(biāo)分量改為該平面上的坐標(biāo)值,其余坐標(biāo)不變。

            8 圖8 正交投影

            比如將點(1,1,5)正交投影到z=0的平面上,那么投影后的坐標(biāo)為(1,1,0)。在openGL中,設(shè)置正交投影可以使用函數(shù):

            1. glOrtho (GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble zNear, GLdouble zFar)  

            該函數(shù)可以設(shè)置正交投影的投影空間,在該空間以外的坐標(biāo)點就不會被投影到投影平面上。函數(shù)中的六個參數(shù)分是投影空間六個平面,如圖9:

            9

            圖9 OpenGL正交投影空間和投影變換

            在圖9中,大的投影空間是根據(jù)這六個參數(shù)設(shè)置的投影空間,OpenGL會自動將該空間歸一化,也就是將該空間或立方體轉(zhuǎn)化為變長為1的正六面體投影 空間,并且該證六面體的中心在相機坐標(biāo)系的原點。一旦設(shè)置使用glortho函數(shù)設(shè)置投影空間,OpenGL會生成投影矩陣。這個矩陣的作用就是將坐標(biāo)進 行正交投影并且將投影后的坐標(biāo)正規(guī)化(轉(zhuǎn)換到-1到1之間)。要注意的是,生成該矩陣的時候,OpenGL會把右手坐標(biāo)系轉(zhuǎn)換為左手坐標(biāo)系。原因很簡單, 右手坐標(biāo)系的Z軸向平面外的,這樣不符合我們的習(xí)慣。該矩陣的矩陣推導(dǎo)這里就不詳細說明了,不了解的同學(xué)可以參考游戲數(shù)學(xué)方面資料,這里只給出正交投影矩 陣。

            15 這個矩陣看來很復(fù)雜,其實計算很簡單。舉個例子,現(xiàn)在設(shè)置了這樣的正交投影空間glOrtho(-10,10,-10,10,-10,10),這是個正六面體空間,變長為10。把這些參數(shù)帶入上面的矩陣可以得到

            gif.latex8 現(xiàn)在還是在OpenGL程序中來檢查一下。在OpenGL程序中添加下面代碼段:

            1. //投影設(shè)置  
            2. glMatrixMode(GL_PROJECTION);  
            3. glLoadIdentity();  
            4. glOrtho(-10,10,-10,10,-10,10);  
            5. glMatrixMode(GL_MODELVIEW);  
            6. glGetFloatv(GL_PROJECTION_MATRIX,m)  
            glGetFloatv(GL_PROJECTION_MATRIX,m)處設(shè)定斷點就可以看到圖10中所顯示的信息。

            10 圖10 正交變換矩陣數(shù)據(jù)

            1.2.2透視投影

            透視投影和正交投影最大的區(qū)別就是透視投影具有遠近感。

            11

            圖11 透視投影

            透視投影采用了圖11中的模型,這樣的模型就是保證遠的物體看起來小,近的物體看起來大。 在OpenGL中設(shè)置透視投影可以使用函數(shù):

            1. void APIENTRY gluPerspective (GLdouble fovy, GLdouble aspect, GLdouble zNear, GLdouble zFar);  

            該函數(shù)也會根據(jù)給定的參數(shù)生成一個投影空間。如圖11中,該投影空間是一個截頭體。同樣地,OpenGL會自動生成透視投影矩陣,該矩陣也會讓3D坐標(biāo)投影在投影平面上,并且將投影后的坐標(biāo)也進行正規(guī)化。下面也直接給出OpenGL中使用的透視投影矩陣。

            16 下面在OpenGL中添加下面代碼段:

            1. //投影設(shè)置  
            2. glMatrixMode(GL_PROJECTION);  
            3. glLoadIdentity();  
            4. gluPerspective(45, 1.0, 1.0, 100);  
            5. glMatrixMode(GL_MODELVIEW);  
            6. glGetFloatv(GL_PROJECTION_MATRIX,m)  

            設(shè)置斷點后,我們可以看到圖12中顯示的數(shù)據(jù)。

            12圖12 透視變換矩陣數(shù)據(jù)

            到此為止,整個投影變換就完成了。透過投影變換后得到的是正規(guī)化的投影平面坐標(biāo)。這為下一個階段的視口變換(View port Transformation)做好了準(zhǔn)備。

            1.3視口變換

            現(xiàn)在到了最后一個階段了。這個階段叫做視口變換,它把上個階段得到的正規(guī)化的投影坐標(biāo)轉(zhuǎn)化為windows 窗口坐標(biāo)。視口變換會將投影平面上的畫面映射到窗口上。在OpenGL中可以使用函數(shù)

            1. GLAPI void GLAPIENTRY glViewport (GLint x, GLint y, GLsizei width, GLsizei height);  

            來進行對窗口的映射,如圖13。

            13
            圖13 視口變換glViewport(width/2, 0, width/2, height/2)

            舉個例子說明,比如上個階段中得到了一個頂點的坐標(biāo)為(0,0,0.5,1),根據(jù)這個坐標(biāo),該頂點位于投影平面的正中間。如果將該點映射到大小為 50*50的窗口上時,那么它應(yīng)該位于屏幕的中間,坐標(biāo)為(25,25, 0.5,1)。當(dāng)然這里深度值0.5是不會改變的。有的同學(xué)肯定有疑問了,既然投影到了窗口上,那么還要深度值0.5干什么?這里要注意的是,雖然在窗口 上顯示時只需要x,y坐標(biāo)就夠了,但是要在2D窗口上顯示3D圖形時深度值是不可少的。這里的深度值不是用于顯示,而是用于在光柵化的時候進行深度測試。

            OpenGL也會根據(jù)glViewport函數(shù)提供的參數(shù)值生成一個視口變換矩陣

            17 該矩陣把上個階段得到的正規(guī)化坐標(biāo)映射到窗口上,并且將正規(guī)化坐標(biāo)中的深度值在轉(zhuǎn)換到0到1之間。所以在深度緩沖中最大值為1,最小值為0。視口變換結(jié)束 后,OpenGL中主要的圖形管線階段就算完成了,后面就是光柵化等等操作。再來回顧一下圖1,現(xiàn)在相信大家對這個渲染管線有了一定的認識了,也明白了每 一個階段對應(yīng)的變換矩陣以及如何進行坐標(biāo)之間的轉(zhuǎn)換的。

            2. 屏幕坐標(biāo)轉(zhuǎn)換為世界坐標(biāo)

            通過前面的教程,以及現(xiàn)在大家對OpenGL整個渲染管線理解后,現(xiàn)在要將屏幕上一點坐標(biāo)轉(zhuǎn)換為世界坐標(biāo)就比較容易了。從圖形管線的開始到結(jié)束,一 個模型坐標(biāo)系中的坐標(biāo)被轉(zhuǎn)化為了屏幕坐標(biāo),那么現(xiàn)在把整個過程倒過來的話,屏幕上一點坐標(biāo)也可以轉(zhuǎn)為為世界坐標(biāo)。只要在對應(yīng)的階段求得對應(yīng)變換矩陣的逆矩 陣,就可以得到前一個階段的坐標(biāo)。這整個過程可以用圖14表示。

            14 圖14屏幕坐標(biāo)轉(zhuǎn)換為世界坐標(biāo)

            圖中顯示的過程完全就是OpenGL渲染管線的逆過程,通過這個過程,屏幕上的點就可以轉(zhuǎn)化為世界坐標(biāo)系中的點了。可能又有的同學(xué)要問,當(dāng)鼠標(biāo)點擊屏幕上一點的時候并沒有深度信息,轉(zhuǎn)換的時候要怎么辦呢?這個時候可以使用OpenGL函數(shù)

            1. void glReadPixels (GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLvoid *pixels);  

            該函數(shù)能夠獲得屏幕上一點對應(yīng)像素的深度信息。有了這個深度信息,就可以利用上面過程把屏幕上一點轉(zhuǎn)換為世界坐標(biāo)了。在OpenGL中,上面的過程其實已經(jīng)有現(xiàn)成的函數(shù)可以使用,那就是

            1. int APIENTRY gluUnProject (  
            2.     GLdouble  winx, GLdouble  winy,  
            3.     GLdouble  winz,  
            4.     const GLdouble modelMatrix[16],  
            5.     const GLdouble projMatrix[16],  
            6.     const GLint    viewport[4],  
            7.     GLdouble  *objx,  GLdouble  *objy,  
            8.     GLdouble       *objz);  

            該函數(shù)直接將屏幕上一點轉(zhuǎn)換對應(yīng)的世界坐標(biāo),該函數(shù)的內(nèi)部實現(xiàn)其實還是上面的那么逆過程。下面給出利用該函數(shù)獲取世界坐標(biāo)的代碼段。

            1. GVector screen2world(int x, int y)  
            2. {  
            3.        GLint viewport[4];  
            4.        GLdouble modelview[16];  
            5.        GLdouble projection[16];  
            6.        GLfloat winX, winY, winZ;  
            7.        GLdouble posX, posY, posZ;  
            8.        glGetDoublev(GL_MODELVIEW_MATRIX, modelview);  
            9.        glGetDoublev(GL_PROJECTION_MATRIX, projection);  
            10.        glGetIntegerv(GL_VIEWPORT, viewport);  
            11.        winX = (float)x;  
            12.        winY = (float)viewport[3] - (float)y;  
            13.        glReadPixels(x, int(winY), 1, 1, GL_DEPTH_COMPONENT, GL_FLOAT, &winZ);  
            14.        gluUnProject(winX, winY, winZ, modelview, projection, viewport, &posX, &posY, &posZ);  
            15.        GVector v(4, posX, posY, posZ, 1.0);  
            16.        return v;  
            17. }  
            代碼中函數(shù)返回類型GVector是用戶定義的向量類,返回的是齊次坐標(biāo)。
            posted on 2012-12-07 20:40 小果子 閱讀(1287) 評論(0)  編輯 收藏 引用 所屬分類: 學(xué)習(xí)筆記游戲開發(fā)
            久久久久亚洲国产| 国产精品久久影院| 久久国产精品无码一区二区三区| 国产精品视频久久| 久久精品极品盛宴观看| 久久精品国产精品青草 | 久久精品国产99国产精品导航| 国产精品久久久久9999| 免费久久人人爽人人爽av| 国产成人久久精品区一区二区| 国产欧美久久久精品影院| 女人香蕉久久**毛片精品| 亚洲精品乱码久久久久66| 午夜精品久久久久久久无码| 狠狠88综合久久久久综合网| 伊人久久精品无码二区麻豆| 日韩欧美亚洲国产精品字幕久久久| 国产精品久久久久久福利漫画| 色综合久久久久无码专区| 亚洲精品国产第一综合99久久| 国内精品伊人久久久久影院对白 | 国产成人久久精品二区三区| 久久精品国产亚洲AV电影| 久久中文字幕视频、最近更新| 91精品婷婷国产综合久久| 国产91色综合久久免费| 久久国产精品77777| 亚洲国产另类久久久精品小说| 免费精品久久天干天干| 久久亚洲熟女cc98cm| 久久无码专区国产精品发布| 亚洲?V乱码久久精品蜜桃 | 国产精品99久久精品爆乳| 久久精品国产99国产精品澳门| 91精品国产高清久久久久久io| 久久精品蜜芽亚洲国产AV| 亚洲精品无码久久久影院相关影片| 久久婷婷是五月综合色狠狠| 亚洲国产成人久久综合碰| 伊色综合久久之综合久久| 久久人人爽人人爽人人片AV不|