• <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>

            天行健 君子當自強而不息

            坐標系與基本圖元(6)

            靈活頂點格式

            靈活頂點格式(Flexible Vertex Format, FVF)用來描述在頂點緩沖區中的頂點存儲格式中包含了哪些屬性。Direct3D應用程序可以用幾種不同的方式定義靈活頂點格式。靈活頂點格式使應用程序只使用它需要的頂點數據,排除那些它不需要的組成成分。這樣,應用程序可以節省內存空間,減少系統帶寬。通過D3DFVF的組合,可以描述圖元頂點的格式。靈活頂點格式指定的格式包括點的大小,用D3DFVF_PSIZE指定,該大小在投影機空間用來表示未經變換的頂點,在設備空間用來表示經過變換的頂點。

            接口IDirect3DDevice9的渲染方法能夠接受這些標志的組合,并用它們來決定如何渲染圖元。這些標志告訴系統應用程序所使用的頂點的組成,包括頂點的位置、頂點混合權重、法向量、顏色和紋理坐標的格式和數量,以及向Direct3D申請何種渲染流水線。另外,提交或撤銷一個具體的頂點格式標志會告訴系統哪些頂點組成單元還留在系統中,哪些已經被忽略了。

            紋理坐標可用不同的格式聲明,紋理可以由一個坐標尋址,也可以由多達4個紋理坐標來尋址。使用宏集合D3DFVF_TEXCOORDSIZEn可以創建位模式,定義要使用的頂點格式所需的紋理坐標格式。

            為了使用索引頂點混合,標志D3DFVF_LASTBETA_UBYTE4應該追加到頂點靈活格式中去。這個標志的出現表明第5個混合權重將被當作一個DWORD值,而不是一個浮點值。

             

            渲染狀態

            設備渲染狀態控制Direct3D設備的光柵化組件的行為。通過改變渲染狀態屬性,可以控制使用何種著色模式,如何進行霧化及其他光柵化操作。Direct3D編程很大一部分工作就是設置合適的渲染狀態。

            Direct3D圖形程序通過調用IDirect3DDevice9::SetRenderState()函數來設置渲染狀態。枚舉類型D3DRENDERSTATETYPE列舉出所有可能的渲染狀態。應用程序將D3DRENDERSTATETYPE類型的某一個枚舉值作為第一個參數傳遞給函數SetRenderState(),然后用第二個參數指定相應的渲染狀態。


            著色模式

            Direct3D中的物體表面是由許許多多的多邊形構成的。當渲染一個物體的多邊形時,不同的著色模式在其表面產生不同的效果。著色模式決定了多邊形上每個點的顏色和光照的強度。Direct3D提供兩種著色模式:平面著色模式(FLAT)和戈勞德著色模式(GOURAUD)。

            (1)平面著色模式

            在平面著色模式下,Direct3D渲染一個多邊形時,把多邊形第一個頂點的顏色作為整個多邊形的顏色進行著色,也就是說,在平面著色模式下,一個多邊形內的所有像素的顏色都等于該多邊形第一個頂點的顏色。如果這些多邊形不共面,用平面著色模式渲染的三維圖形將出現明顯的陡沿。平面著色模式是渲染速度最快的著色模式。

            (2)戈勞德著色模式

            當用戈勞德著色渲染一個多邊形時,它會先用頂點法向量和燈光參數計算每個頂點的顏色。然后,在該多邊形的表面上進行線性插值,進而得到每個像素的顏色。例如,頂點1顏色的紅色值為0.8,頂點2顏色的紅色至為0.4,使用戈勞德著色模式和RGB顏色模式,Direct3D燈光組件將使這兩點連線上中點的顏色的紅色值為0.6。

            (3)著色模式比較

            在平面著色模式中,相鄰兩個面之間會有明顯的邊緣,而采用戈勞德著色模式時,邊緣處的著色值會由內插運算產生,因而最后會得到一個彎曲的表面。在戈勞德著色模式下,對平面光照的處理要比在平面著色模式下更真實。在平面著色模式下,一個面的顏色是唯一的,但戈勞德著色模式可以使光線更準確地照射在每一個面上。當一個表面距離一個點光源很近時,它們的區別將會更明顯地表現出來。

            戈勞德著色模式可以使在平面著色模式下多邊形間的陡沿變得平滑。然而這樣可能會導致馬赫帶效應(Mach bands)的產生,也就是相鄰的顏色或光線帶之間不能很平滑的相互融合。對于程序開發人員來說,可以通過增加構成對象的多邊形的數目來降低馬赫帶效應,當然也可以通過提高屏幕分辨率或者增加程序的顏色深度來達到目的。

            (4)設置著色模式

            Direct3D一次只能選擇一種著色模式。默認情況下,選擇戈勞德著色模式。在Direct3D圖形程序中,調用IDirect3DDevice9:: SetRenderState()方法來改變著色模式。設置狀態參數為D3DRS_SHADEMODE,該狀態參數值必須是D3DSHADEMODE枚舉成員的一個。

            Defines constants that describe the supported shading modes.

            typedef enum D3DSHADEMODE
            {
            D3DSHADE_FLAT = 1,
            D3DSHADE_GOURAUD = 2,
            D3DSHADE_PHONG = 3,
            D3DSHADE_FORCE_DWORD = 0x7fffffff,
            } D3DSHADEMODE, *LPD3DSHADEMODE;

            Constants

            D3DSHADE_FLAT
            Flat shading mode. The color and specular component of the first vertex in the triangle are used to determine the color and specular component of the face. These colors remain constant across the triangle; that is, they are not interpolated. The specular alpha is interpolated. See Remarks.
            D3DSHADE_GOURAUD
            Gouraud shading mode. The color and specular components of the face are determined by a linear interpolation between all three of the triangle's vertices.
            D3DSHADE_PHONG
            Not supported.
            D3DSHADE_FORCE_DWORD
            Forces this enumeration to compile to 32 bits in size. Without this value, some compilers would allow this enumeration to compile to a size other than 32 bits. This value is not used.

            Remarks

            The first vertex of a triangle for flat shading mode is defined in the following manner.

            • For a triangle list, the first vertex of the triangle i is i * 3.
            • For a triangle strip, the first vertex of the triangle i is vertex i.
            • For a triangle fan, the first vertex of the triangle i is vertex i + 1.

            The members of this enumerated type define the vales for the D3DRS_SHADEMODE render state.

             

            多邊形填充模式

            默認狀態下,Direct3D會把渲染好的多邊形面的圖像繪制出來,這種方法適合于絕大多數情況。然而,有時為了調試程序或其他特殊目的,可能只需繪制出多邊形的頂點或邊。這可以通過設置渲染狀態D3DRS_FILLMODE來實現,通過為D3DRS_FILLMODE渲染狀態指定枚舉類型D3DFILLMODE中的一個數值,來選擇多邊形填充模式。

            Defines constants describing the fill mode.

            typedef enum D3DFILLMODE
            {
            D3DFILL_POINT = 1,
            D3DFILL_WIREFRAME = 2,
            D3DFILL_SOLID = 3,
            D3DFILL_FORCE_DWORD = 0x7fffffff,
            } D3DFILLMODE, *LPD3DFILLMODE;

            Constants

            D3DFILL_POINT
            Fill points.
            D3DFILL_WIREFRAME
            Fill wireframes.
            D3DFILL_SOLID
            Fill solids.
            D3DFILL_FORCE_DWORD
            Forces this enumeration to compile to 32 bits in size. Without this value, some compilers would allow this enumeration to compile to a size other than 32 bits. This value is not used.

            Remarks

            The values in this enumerated type are used by the D3DRS_FILLMODE render state.

             

            著色模式和填充模式示例程序

            示例程序RenderState演示了著色模式和填充模式對圖形顯示結果的影響,在RenderState示例程序中用全局變量g_is_flat和g_fill_mode控制渲染圖形使用的著色模式和填充模式,通過單擊鼠標左鍵在兩種著色模式之間進行切換,單擊鼠標右鍵在三種填充模式之間進行切換。

            Gouraud著色模式

             

            FLAT著色模式

             

            點模式

             

            線模式

             

            完整源代碼:

            #include <d3d9.h>

            #define CLASS_NAME    "GameApp"

            #define release_com(p)    do { if(p) { (p)->Release(); (p) = NULL; } } while(0)

            IDirect3D9
            *                g_d3d;
            IDirect3DDevice9
            *        g_device;
            IDirect3DVertexBuffer9
            * g_vertex_buffer;
            bool                    g_is_flat    = false;
            DWORD                    g_fill_mode 
            = D3DFILL_SOLID;

            struct sCustomVertex
            {
                
            float x, y, z, rhw;
                DWORD color;
            };

            #define D3DFVF_CUSTOM_VERTEX (D3DFVF_XYZRHW | D3DFVF_DIFFUSE) 

            void init_vertices()
            {
                sCustomVertex vertices[] 
            =
                {
                    { 
            100.0f400.0f0.5f1.0f0xffff0000, },
                    { 
            300.0f,  50.0f0.5f1.0f0xff00ff00, },         
                    { 
            500.0f400.0f0.5f1.0f0xff0000ff, },
                };

                
            // push vertex data into vertex buffer

                g_device
            ->CreateVertexBuffer(sizeof(vertices), 0, D3DFVF_CUSTOM_VERTEX, D3DPOOL_DEFAULT, &g_vertex_buffer, NULL);

                
            void* ptr;

                g_vertex_buffer
            ->Lock(0sizeof(vertices), (void**)&ptr, 0);
                memcpy(ptr, vertices, 
            sizeof(vertices));
                g_vertex_buffer
            ->Unlock();
            }

            bool init_d3d(HWND hwnd)
            {
                g_d3d 
            = Direct3DCreate9(D3D_SDK_VERSION);

                
            if(g_d3d == NULL)
                    
            return false;

                D3DPRESENT_PARAMETERS d3dpp;
                ZeroMemory(
            &d3dpp, sizeof(d3dpp));

                d3dpp.Windowed            
            = TRUE;
                d3dpp.SwapEffect        
            = D3DSWAPEFFECT_DISCARD;
                d3dpp.BackBufferFormat    
            = D3DFMT_UNKNOWN;

                
            if(FAILED(g_d3d->CreateDevice(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, hwnd, D3DCREATE_SOFTWARE_VERTEXPROCESSING,
                                              
            &d3dpp, &g_device)))
                {
                    
            return false;
                }

                init_vertices();

                
            return true;
            }

            void cleanup()
            {    
                release_com(g_vertex_buffer);
                release_com(g_device);
                release_com(g_d3d);
            }

            void render()
            {
                g_device
            ->SetRenderState(D3DRS_SHADEMODE, g_is_flat ? D3DSHADE_FLAT : D3DSHADE_GOURAUD);
                g_device
            ->SetRenderState(D3DRS_FILLMODE,  g_fill_mode);

                g_device
            ->Clear(0, NULL, D3DCLEAR_TARGET, D3DCOLOR_XRGB(555), 1.0f0);    

                g_device
            ->BeginScene();

                g_device
            ->SetStreamSource(0, g_vertex_buffer, 0sizeof(sCustomVertex));
                g_device
            ->SetFVF(D3DFVF_CUSTOM_VERTEX);    
                g_device
            ->DrawPrimitive(D3DPT_TRIANGLELIST, 01);

                g_device
            ->EndScene();

                g_device
            ->Present(NULL, NULL, NULL, NULL);
            }

            LRESULT WINAPI WinProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
            {
                
            switch(msg)
                {
                
            case WM_LBUTTONDOWN:
                    g_is_flat 
            = !g_is_flat;
                    
            break;

                
            case WM_RBUTTONDOWN:
                    
            if(g_fill_mode == D3DFILL_POINT)
                        g_fill_mode 
            = D3DFILL_WIREFRAME;
                    
            else if(g_fill_mode == D3DFILL_WIREFRAME)
                        g_fill_mode 
            = D3DFILL_SOLID;
                    
            else if(g_fill_mode == D3DFILL_SOLID)
                        g_fill_mode 
            = D3DFILL_POINT;

                    
            break;

                
            case WM_KEYDOWN:
                    
            switch(wParam)
                    {    
                    
            case VK_ESCAPE:
                        DestroyWindow(hwnd);
                        
            break;
                    }    
                    
            break;

                
            case WM_DESTROY:        
                    PostQuitMessage(
            0);
                    
            return 0;
                }

                
            return DefWindowProc(hwnd, msg, wParam, lParam);
            }

            int WINAPI WinMain(HINSTANCE inst, HINSTANCE, LPSTR, INT)
            {
                WNDCLASSEX wc;

                wc.cbSize            
            = sizeof(WNDCLASSEX);
                wc.style            
            = CS_CLASSDC;
                wc.lpfnWndProc        
            = WinProc;
                wc.cbClsExtra        
            = 0;
                wc.cbWndExtra        
            = 0;
                wc.hInstance        
            = inst;
                wc.hIcon            
            = NULL;
                wc.hCursor            
            = NULL;
                wc.hbrBackground    
            = NULL;
                wc.lpszMenuName        
            = NULL;
                wc.lpszClassName    
            = CLASS_NAME;
                wc.hIconSm            
            = NULL;

                
            if(! RegisterClassEx(&wc))
                    
            return -1;

                HWND hwnd 
            = CreateWindow(CLASS_NAME, "Direct3D App", WS_OVERLAPPEDWINDOW, 200100600500,
                                         NULL, NULL, wc.hInstance, NULL);

                
            if(hwnd == NULL)
                    
            return -1;

                
            if(init_d3d(hwnd))
                {
                    ShowWindow(hwnd, SW_SHOWDEFAULT);
                    UpdateWindow(hwnd);

                    MSG msg;
                    ZeroMemory(
            &msg, sizeof(msg));

                    
            while(msg.message != WM_QUIT)
                    {
                        
            if(PeekMessage(&msg, NULL, 00, PM_REMOVE))
                        {
                            TranslateMessage(
            &msg);
                            DispatchMessage(
            &msg);
                        }
                            
                        render();
                    }
                }

                cleanup();

                UnregisterClass(CLASS_NAME, wc.hInstance);    

                
            return 0;
            }

             

            posted on 2008-05-01 10:17 lovedday 閱讀(1504) 評論(1)  編輯 收藏 引用

            評論

            # re: 坐標系與基本圖元(6) 2008-07-24 00:16 riki

            很黃很暴力,希望能認識志同道合的朋友  回復  更多評論   

            公告

            導航

            統計

            常用鏈接

            隨筆分類(178)

            3D游戲編程相關鏈接

            搜索

            最新評論

            97精品国产91久久久久久| 久久久久久A亚洲欧洲AV冫| 中文字幕无码久久久| 亚洲综合久久久| 欧美黑人激情性久久| 91久久精一区二区三区大全| 久久99精品国产麻豆不卡| 精品久久久久久国产| 久久最近最新中文字幕大全| 日韩美女18网站久久精品| 亚洲伊人久久大香线蕉综合图片| 成人综合伊人五月婷久久| 久久强奷乱码老熟女网站| 蜜臀av性久久久久蜜臀aⅴ | 亚洲人成网站999久久久综合| 无码人妻精品一区二区三区久久| 国産精品久久久久久久| 久久不见久久见免费视频7| 亚洲&#228;v永久无码精品天堂久久 | 亚洲欧美日韩久久精品第一区| 99久久成人18免费网站| 五月丁香综合激情六月久久| 久久国产综合精品五月天| 久久久久久久亚洲Av无码| 中文成人无码精品久久久不卡| 日韩精品国产自在久久现线拍| 亚洲狠狠婷婷综合久久蜜芽| 四虎久久影院| 久久久久久极精品久久久| 国产∨亚洲V天堂无码久久久| 国产毛片欧美毛片久久久| 四虎亚洲国产成人久久精品| 久久久WWW免费人成精品| 国内精品久久久久久久涩爱| 久久婷婷综合中文字幕| 青青青伊人色综合久久| 久久免费精品视频| 品成人欧美大片久久国产欧美...| 国产精品毛片久久久久久久| 国产午夜免费高清久久影院| 久久久久久毛片免费播放|