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

            天行健 君子當自強而不息

            網格模型高級技術(12)

            為了方便加載.x文件中的框架和網格模型數據,Direct3D提供了一個ID3DXAllocateHierarchy接口,該接口中有4個純虛函數:用來創建框架的CreateFrame(),創建網格容器的CreateMeshContainer(),銷毀框架的DestroyFrame(),銷毀網格容器的DestroyMeshContainer()。應用程序會在相應的時機自動調用這些對應的函數,以構建或者銷毀對應的框架或網格模型。

            This interface is implemented by the application to allocate or free frame and mesh container objects. Methods on this are called during loading and destroying frame hierarchies.

            Method Description
            ID3DXAllocateHierarchy::CreateFrame Requests allocation of a frame object.
            ID3DXAllocateHierarchy::CreateMeshContainer Requests allocation of a mesh container object.
            ID3DXAllocateHierarchy::DestroyFrame Requests deallocation of a frame object.
            ID3DXAllocateHierarchy::DestroyMeshContainer Requests deallocation of a mesh container object.

             

            ID3DXAllocateHierarchy::CreateFrame

            Requests allocation of a frame object.

            HRESULT CreateFrame(
            LPCSTR Name,
            LPD3DXFRAME * ppNewFrame
            );

            Parameters

            Name
            [in] Name of the frame to be created.
            ppNewFrame
            [out, retval] Returns the created frame object.

            Return Values

            The return values of this method are implemented by an application programmer. In general, if no error occurs, program the method to return D3D_OK. Otherwise, program the method to return an appropriate error message from D3DERR or D3DXERR, as this will cause D3DXLoadMeshHierarchyFromX to fail also, and return the error.

             

            ID3DXAllocateHierarchy::CreateMeshContainer

            Requests allocation of a mesh container object.

            HRESULT CreateMeshContainer(
            LPCSTR Name,
            CONST D3DXMESHDATA * pMeshData,
            CONST D3DXMATERIAL * pMaterials,
            CONST D3DXEFFECTINSTANCE * pEffectInstances,
            DWORD NumMaterials,
            CONST DWORD * pAdjacency,
            LPD3DXSKININFO pSkinInfo,
            LPD3DXMESHCONTAINER * ppNewMeshContainer
            );

            Parameters

            Name
            [in] Name of the mesh.
            pMeshData
            [in] Pointer to the mesh data structure.
            pMaterials
            [in] Array of materials used in the mesh.
            pEffectInstances
            [in] Array of effect instances used in the mesh. 
            NumMaterials
            [in] Number of materials in the materials array.
            pAdjacency
            [in] Adjacency array for the mesh.
            pSkinInfo
            [in] Pointer to the skin mesh object if skin data is found.
            ppNewMeshContainer
            [out, retval] Returns the created mesh container.

            Return Values

            The return values of this method are implemented by an application programmer. In general, if no error occurs, program the method to return D3D_OK. Otherwise, program the method to return an appropriate error message from D3DERR or D3DXERR, as this will cause D3DXLoadMeshHierarchyFromX to fail also, and return the error.

             

            ID3DXAllocateHierarchy::DestroyFrame

            Requests deallocation of a frame object.

            HRESULT DestroyFrame(
            LPD3DXFRAME pFrameToFree
            );

            Parameters

            pFrameToFree
            [in] Pointer to the frame to be deallocated.

            Return Values

            The return values of this method are implemented by an application programmer. In general, if no error occurs, program the method to return D3D_OK. Otherwise, program the method to return an appropriate error message from D3DERR or D3DXERR, as this will cause D3DXLoadMeshHierarchyFromX to fail also, and return the error.

             

            ID3DXAllocateHierarchy::DestroyMeshContainer

            Requests deallocation of a mesh container object.

            HRESULT DestroyMeshContainer(
            LPD3DXMESHCONTAINER pMeshContainerToFree
            );

            Parameters

            pMeshContainerToFree
            [in] Pointer to the mesh container object to be deallocated.

            Return Values

            The return values of this method are implemented by an application programmer. In general, if no error occurs, program the method to return D3D_OK. Otherwise, program the method to return an appropriate error message from D3DERR or D3DXERR, as this will cause D3DXLoadMeshHierarchyFromX to fail also, and return the error.

             

            cAllocateHierarchy類繼承自ID3DXAllocateHierarchy接口,在cAllocateHierarchy類需要重載這4個純虛函數以實現動畫網格模型數據的加載和釋放。

            該類的定義如下:

            class cAllocateHierarchy : public ID3DXAllocateHierarchy
            {
            private:
            HRESULT AllocateName(LPCSTR name, LPSTR* ret_name);
            public:
            STDMETHOD(CreateFrame)(LPCSTR name, LPD3DXFRAME* ret_frame);
            	STDMETHOD(CreateMeshContainer)(LPCSTR name, 
            CONST D3DXMESHDATA* mesh_data,
            CONST D3DXMATERIAL* xmaterials,
            CONST D3DXEFFECTINSTANCE* effect_instances,
            DWORD num_materials,
            CONST DWORD* adjacency,
            LPD3DXSKININFO skin_info,
            LPD3DXMESHCONTAINER* ret_mesh_container);
                STDMETHOD(DestroyFrame)(THIS_ LPD3DXFRAME frame_to_free);
            STDMETHOD(DestroyMeshContainer)(THIS_ LPD3DXMESHCONTAINER base_mesh_container);
            };
             

            函數AllocateFrame()用來為框架生成一個名稱:

            HRESULT cAllocateHierarchy::AllocateName(LPCSTR name, LPSTR* ret_name)
            {
            if(name != NULL)
            {
            UINT length = (UINT)strlen(name) + 1;
            		*ret_name = new CHAR[length];
            		if(*ret_name == NULL)
            return E_OUTOFMEMORY;
            		memcpy(*ret_name, name, length * sizeof(CHAR));
            }
            else
            {
            *ret_name = NULL;
            }
            	return S_OK;
            }

             

            函數CreateFrame()的作用在于生成一個新的擴展框架,并按照指定的參數為該框架命名:

            HRESULT cAllocateHierarchy::CreateFrame(LPCSTR name, LPD3DXFRAME* ret_frame)
            {
            *ret_frame = NULL;
            	D3DXFRAME_DERIVED* new_frame = new D3DXFRAME_DERIVED;
            	if(new_frame == NULL)
            return E_OUTOFMEMORY;
            	HRESULT hr = AllocateName(name, &new_frame->Name);
            	if(FAILED(hr))
            {
            delete new_frame;
            return hr;
            }
            	D3DXMatrixIdentity(&new_frame->TransformationMatrix);
            D3DXMatrixIdentity(&new_frame->CombinedTransformMatrix);
            	new_frame->pMeshContainer   = NULL;
            new_frame->pFrameSibling = NULL;
            new_frame->pFrameFirstChild = NULL;
            	*ret_frame = new_frame;
            	return S_OK;
            }

             

            在一個框架創建好后,需要創建該框架的網格容器,這通過函數CreateMeshContainer()來實現:

            HRESULT cAllocateHierarchy::CreateMeshContainer(LPCSTR name, 
            CONST D3DXMESHDATA* mesh_data,
            CONST D3DXMATERIAL* xmaterials,
            CONST D3DXEFFECTINSTANCE* effect_instances,
            DWORD num_materials,
            CONST DWORD* adjacency,
            LPD3DXSKININFO skin_info,
            LPD3DXMESHCONTAINER* ret_mesh_container)
            {
            *ret_mesh_container = NULL;
            	if(mesh_data->Type != D3DXMESHTYPE_MESH)
            return E_FAIL;
            	ID3DXMesh* mesh_ptr = mesh_data->pMesh;
            DWORD fvf = mesh_ptr->GetFVF();
            	if(fvf == 0)
            return E_FAIL;
            	// create a mesh container and zero it
            	D3DXMESHCONTAINER_DERIVED* new_mesh_container = new D3DXMESHCONTAINER_DERIVED;
            	if(new_mesh_container == NULL)
            return E_OUTOFMEMORY;
            	memset(new_mesh_container, 0, sizeof(D3DXMESHCONTAINER_DERIVED));
            	// copy mesh name
            	HRESULT hr = AllocateName(name, &new_mesh_container->Name);
            	if(FAILED(hr))
            {
            DestroyMeshContainer(new_mesh_container);
            return hr;
            }
            	IDirect3DDevice9* device;
            mesh_ptr->GetDevice(&device);
            	new_mesh_container->MeshData.Type = D3DXMESHTYPE_MESH;
            	// be sure mesh contain normal
            if(!(fvf & D3DFVF_NORMAL))
            {
            hr = mesh_ptr->CloneMeshFVF(mesh_ptr->GetOptions(), fvf | D3DFVF_NORMAL, device,
            &new_mesh_container->MeshData.pMesh);
            		if(FAILED(hr))
            {
            release_com(device);
            DestroyMeshContainer(new_mesh_container);
            return hr;
            }
            		mesh_ptr = new_mesh_container->MeshData.pMesh;
            D3DXComputeNormals(mesh_ptr, NULL);
            }
            else
            {
            new_mesh_container->MeshData.pMesh = mesh_ptr;
            mesh_ptr->AddRef(); // !! important, so DestroyMeshContainer() will not crash.
            }
            	// load materials and textures
            	UINT num_faces = mesh_ptr->GetNumFaces();
            	new_mesh_container->NumMaterials = max(1, num_materials);
            new_mesh_container->pMaterials = new D3DXMATERIAL[new_mesh_container->NumMaterials];
            new_mesh_container->ppTextures = new LPDIRECT3DTEXTURE9[new_mesh_container->NumMaterials];
            new_mesh_container->pAdjacency = new DWORD[num_faces * 3];
            	if((new_mesh_container->pAdjacency == NULL) || (new_mesh_container->pMaterials == NULL) ||
            (new_mesh_container->ppTextures == NULL))
            {
            release_com(device);
            DestroyMeshContainer(new_mesh_container);
            return E_OUTOFMEMORY;
            }
            	memcpy(new_mesh_container->pAdjacency, adjacency, sizeof(DWORD) * num_faces * 3);
            memset(new_mesh_container->ppTextures, 0, sizeof(LPDIRECT3DTEXTURE9) * new_mesh_container->NumMaterials);
            	D3DXMATERIAL* xmaterials_ptr = new_mesh_container->pMaterials;
            	if(num_materials > 0)
            {
            memcpy(xmaterials_ptr, xmaterials, sizeof(D3DXMATERIAL) * num_materials);
            xmaterials_ptr->MatD3D.Ambient = xmaterials_ptr->MatD3D.Diffuse;
            		for(UINT i = 0; i < num_materials; i++)
            {
            if(xmaterials_ptr[i].pTextureFilename != NULL)
            {
            WCHAR w_texture_path[MAX_PATH];
            WCHAR w_filename[MAX_PATH];
            				RemovePathFromFileName(xmaterials_ptr[i].pTextureFilename, w_filename);
            DXUTFindDXSDKMediaFileCch(w_texture_path, MAX_PATH, w_filename);
            				if(FAILED( D3DXCreateTextureFromFileW(device, w_texture_path, &new_mesh_container->ppTextures[i]) ))
            new_mesh_container->ppTextures[i] = NULL;
            }
            }
            }
            else
            {
            xmaterials_ptr[0].pTextureFilename = NULL;
            memset(&xmaterials_ptr[0].MatD3D, 0, sizeof(D3DMATERIAL9));
            		xmaterials_ptr[0].MatD3D.Diffuse.r = 0.5f;
            xmaterials_ptr[0].MatD3D.Diffuse.r = 0.5f;
            xmaterials_ptr[0].MatD3D.Diffuse.b = 0.5f;
            xmaterials_ptr[0].MatD3D.Specular = xmaterials_ptr[0].MatD3D.Diffuse;
            xmaterials_ptr[0].MatD3D.Ambient = xmaterials_ptr[0].MatD3D.Diffuse;
            }
            	release_com(device);
            	*ret_mesh_container = new_mesh_container;
            	return S_OK;
            }

            在此實現的骨骼動畫網格模型中沒有涉及到蒙皮信息,所以在CreateMeshContainer()函數中沒有處理參數skin_info指向的蒙皮信息。

             

            函數DestroyFrame()只有一個參數指向準備釋放的框架對象:

            HRESULT cAllocateHierarchy::DestroyFrame(LPD3DXFRAME frame_to_free) 
            {
            SAFE_DELETE_ARRAY(frame_to_free->Name);
            SAFE_DELETE(frame_to_free);
            	return S_OK;
            }

             

            函數DestroyMeshContainer()也只有一個參數指向將要釋放的網格容器對象:

            HRESULT cAllocateHierarchy::DestroyMeshContainer(LPD3DXMESHCONTAINER base_mesh_container)
            {
            if(base_mesh_container == NULL)
            return S_OK;
            	D3DXMESHCONTAINER_DERIVED* mesh_container = (D3DXMESHCONTAINER_DERIVED*) base_mesh_container;
            	SAFE_DELETE_ARRAY(mesh_container->Name);
            SAFE_DELETE_ARRAY(mesh_container->pAdjacency);
            SAFE_DELETE_ARRAY(mesh_container->pMaterials);
            	if(mesh_container->ppTextures != NULL)
            {
            for(UINT i = 0; i < mesh_container->NumMaterials; i++)
            release_com(mesh_container->ppTextures[i]);
            }
            	SAFE_DELETE_ARRAY(mesh_container->ppTextures);
            	SAFE_RELEASE(mesh_container->MeshData.pMesh);
            SAFE_RELEASE(mesh_container->pSkinInfo);
            	SAFE_DELETE(mesh_container);
            	return S_OK;
            }

            posted on 2008-06-11 14:56 lovedday 閱讀(2173) 評論(3)  編輯 收藏 引用

            評論

            # re: 網格模型高級技術(12) 2010-04-04 11:58 chuckey

            誰寫的這種文章,也太無聊了,全就抄了一下.  回復  更多評論   

            # re: 網格模型高級技術(12) 2010-07-14 14:52 騾子寶

            博主能出來解釋一下嗎?看得云里霧里的。
            FRAME是每個骨骼創建一個FRAME嗎?每個FRMAE都創建一個CONTAINER嗎?  回復  更多評論   

            # re: 網格模型高級技術(12) 2011-05-22 13:04

            看看  回復  更多評論   

            公告

            導航

            統計

            常用鏈接

            隨筆分類(178)

            3D游戲編程相關鏈接

            搜索

            最新評論

            精品无码久久久久久久动漫| 久久久国产精品网站| 亚洲欧美另类日本久久国产真实乱对白| 久久91精品国产91久久麻豆| 精品久久久久久无码中文野结衣| 亚洲国产精品成人久久蜜臀| 浪潮AV色综合久久天堂| 精品视频久久久久| 漂亮人妻被黑人久久精品| 精品久久久久久无码人妻热| 一本色道久久99一综合| 久久亚洲AV无码西西人体| 久久99国产乱子伦精品免费| 久久综合日本熟妇| 色综合合久久天天综合绕视看| 天天综合久久一二三区| 精品999久久久久久中文字幕| 久久国产成人亚洲精品影院| 高清免费久久午夜精品| 久久久久久久久久久精品尤物| 国产叼嘿久久精品久久| 久久精品国产91久久综合麻豆自制 | 国产精品九九九久久九九| 久久人人爽人爽人人爽av| 91久久精品国产91性色也| 无码专区久久综合久中文字幕 | 一本色道久久综合狠狠躁| 日本亚洲色大成网站WWW久久| 久久久久国产一级毛片高清版| 亚洲人成精品久久久久| 久久夜色精品国产噜噜亚洲a| 国产叼嘿久久精品久久| 成人午夜精品久久久久久久小说| 国产精品久久久久久吹潮| 久久婷婷五月综合国产尤物app| 久久精品国产亚洲AV影院| 欧美亚洲国产精品久久久久| 一本色道久久88综合日韩精品 | 99热成人精品免费久久| 91精品国产高清久久久久久国产嫩草 | 久久久久亚洲AV无码观看|