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

            天行健 君子當(dāng)自強(qiáng)而不息

            Working with skeletal animation(7)

            Updating the Skinned Mesh

            When your skeletal structure is in the pose you desire, it's time to update (or rebuild) the skinned mesh to match. Before you rebuild the skinned mesh, you must make sure you have constructed the secondary mesh container and updated the frame hierarchy. To review how to construct the mesh container, consult the "Creating a Secondary Mesh Container" section earlier in this chapter. To refresh your memory about how to update the frame hierarchy, review the "Updating the Hierarchy" section earlier in this chapter. After you're sure of these two things, you can continue.

            To update the skinned mesh, you must first lock the vertex buffers of the skinned mesh and the secondary mesh. This is critical because DirectX will pull the vertex data from the skinned mesh object, apply the bone transformations, and write the resulting vertex data to the secondary mesh object.

            First, though, you need to copy the transformations from the frames to the array of matrices(pBoneMatrices) stored in the mesh container. At the same time, you have to combine the transformations  with the bones' inversed transformations. The inversed bone transformations are responsible for moving the mesh's vertices to the origin of the mesh before you apply the actual transformation. To better understand this, take a look at Figure 4.4

            The mesh in Figure 4.4 is composed of three bones (frames) and a number of vertices. To apply a transformation to any frame, you must move the vertices belonging to the frame to the origin and then apply the transformations.

            You move the vertices around the origin of the mesh before you apply a transformation because a rotation matrix simply rotates vertices around an origin. If you were to rotate a vertex belonging to any bone, the vertex would rotate around the origin of the mesh instead of the bone's joint. For example, if your body was a mesh and you bent your elbow, the vertices constructing your arm's mesh would rotate around your elbow, not the center of your body. After the vertices are moved to the center of the mesh, the transformation is applied (thus rotating the vertices to match the rotation of the bone) and finally translated into position.

            Normally, these inversed bone transformations are stored in the .X file by the 3D modeler used to create the meshes. If you don't have access to this information from an .X file, you can compute it yourself by first updating the frame hierarchy, and then inverting each frame's combined transformation using the D3DXMatrixInverse function. Here's a quick example.

            // pRoot = root D3DXFRAME_EX object
            // pMesh = D3DXMESHCONTAINER_EX object w/mesh data
            // Update the frame hierarchy
            pRoot−>UpdateHierarchy();
            // Go through each bone and calculate the inverse
            for(DWORD i=0;i<NumBones;i++)
            {
            // Grab the transformation using the bone matrix
            D3DXMATRIX matBone = (*pMesh−>ppFrameMatrices);
            	// Invert the matrix
            D3DXMatrixInverse(&matBone, NULL, &matBone);
            	// Store the inversed bone transformation somewhere
            }

            Instead of going through all the trouble of calculating the inversed bone transformations yourself, however, you can rely on the skinned mesh object to supply that information. By calling ID3DXSkinInfo::GetBoneOffsetMatrix, you'll get the inversed bone transformation matrix pointer. Multiply this matrix by a frame transformation matrix, and you're set!

            Using what you just learned, iterate through all the bones, grab the inversed bone transformation, combine it with the frame transformation, and store the result in the pBoneMatrices array.

            for(DWORD i=0;i<pSkinInfo−>GetNumBones();i++) 
            {
            // Set the inversed bone transformation
            pMesh−>pBoneMatrices[i]=(*pSkinInfo−>GetBoneOffsetMatrix(i));
            	// Apply frame transformation
            if(pMesh−>ppFrameMatrices[i])
            pMesh−>pBoneMatrices[i] *= (*pMesh−>ppFrameMatrices[i]);
            }

            Now that you've copied the bones' transformations into the pBoneMatrices array, you can move on to updating the skinned mesh by first locking the vertex buffers for the skinned mesh and the secondary mesh.

            // pSkinMesh = skinned mesh container
            // pMesh = secondary mesh container

            // Lock the meshes' vertex buffers
            void *SrcPtr, *DestPtr;

            pSkinMesh−>LockVertexBuffer(D3DLOCK_READONLY,(void**)&SrcPtr);
            pMesh−>LockVertexBuffer(0, (void**)&DestPtr);

            After you lock the vertex buffers, you need to perform a call to ID3DXSkinInfo::UpdateSkinnedMesh to apply all the bones' transformations to the vertices and write the resulting data to the secondary mesh container.

            Applies software skinning to the target vertices based on the current matrices.

            HRESULT UpdateSkinnedMesh(
            CONST D3DXMATRIX * pBoneTransforms,
            CONST D3DXMATRIX * pBoneInvTransposeTransforms,
            LPCVOID pVerticesSrc,
            PVOID pVerticesDst
            );

            Parameters

            pBoneTransforms
            [in] Bone transform matrix.
            pBoneInvTransposeTransforms
            [in] Inverse transpose of the bone transform matrix.
            pVerticesSrc
            [in] Pointer to the buffer containing the source vertices.
            pVerticesDst
            [in] Pointer to the buffer containing the destination vertices.

            Return Values

            If the method succeeds, the return value is D3D_OK. If the method fails, the return value can be D3DERR_INVALIDCALL.

            Remarks

            When used to skin vertices with two position elements, this method skins the second position element with the inverse of the bone instead of the bone itself.

            To finish, you simply unlock the vertex buffers, and you're ready to render!

            // pSkinInfo = skinned mesh info object

            // Update the skinned mesh using provided transformations
            pSkinInfo−>UpdateSkinnedMesh(pBoneMatrices, NULL, SrcPtr, DestPtr);

            // Unlock the meshes vertex buffers
            pSkinMesh−>UnlockVertexBuffer();
            pMesh−>UnlockVertexBuffer();

             

            Rendering the Skinned Mesh

            Now comes the good part−rendering your secondary mesh and showing the world what it's like to play with powerthe power of skeletal animation and skinned meshes, that is. You only need to depend on the typical mesh−rendering functions to render the secondary mesh. Loop through each material, set the material and texture, and call the ID3DXMesh::DrawSubset function. Loop and continue until all of the subsets have been drawn.

            // pMesh = D3DXMESHCONTAINER_EX object with material data
            // pMeshToDraw = secondary mesh pointer to render
            for(DWORD i=0;i<pMesh−>NumMaterials;i++)
            {
            // Set material and texture
            pD3DDevice−>SetMaterial(&pMesh−>pMaterials[i].MatD3D);
            pD3DDevice−>SetTexture(0, pMesh−>pTextures[i]);
            	// Draw the mesh subset
            pMeshToDraw−>DrawSubset(i);
            }

            posted on 2008-04-23 20:06 lovedday 閱讀(464) 評論(0)  編輯 收藏 引用


            只有注冊用戶登錄后才能發(fā)表評論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            公告

            導(dǎo)航

            統(tǒng)計(jì)

            常用鏈接

            隨筆分類(178)

            3D游戲編程相關(guān)鏈接

            搜索

            最新評論

            色婷婷综合久久久久中文| 亚洲性久久久影院| 66精品综合久久久久久久| 99久久婷婷国产综合精品草原 | 久久成人18免费网站| 热RE99久久精品国产66热| 久久久无码一区二区三区| 久久影视国产亚洲| 久久综合丝袜日本网| 午夜不卡久久精品无码免费| 国产亚洲色婷婷久久99精品91| A级毛片无码久久精品免费 | 久久天天躁夜夜躁狠狠| 品成人欧美大片久久国产欧美... 品成人欧美大片久久国产欧美 | 色综合久久天天综合| 亚洲欧洲日产国码无码久久99| 国产女人aaa级久久久级| 精品免费久久久久久久| 97精品伊人久久久大香线蕉| 久久久久这里只有精品| 国产视频久久| 国产精品久久久久乳精品爆| 久久精品国产91久久麻豆自制| 亚洲精品无码久久千人斩| 久久久黄色大片| 一本色道久久综合| 久久午夜免费视频| 午夜视频久久久久一区| 久久久亚洲精品蜜桃臀| 大蕉久久伊人中文字幕| 91久久九九无码成人网站| 色噜噜狠狠先锋影音久久| 99久久精品国内| 93精91精品国产综合久久香蕉| 久久r热这里有精品视频| 97久久精品人妻人人搡人人玩 | 久久精品人人槡人妻人人玩AV| 婷婷伊人久久大香线蕉AV| 久久久久无码精品国产不卡| 99re久久精品国产首页2020| 91精品国产91久久久久久蜜臀|