VC+DirectShow对视频进行图片处理之五

发表于:2007-06-17来源:作者:点击数: 标签:
应用到D3D中去 平面的图像、影片看的多了,我们不妨到3D 环境中看看影片。 我不会在这里介绍 D3D,您要学习它就得自己找资料,这里只是讲在 3D 环境中播放的关键—— 图片到纹理。 3D 纹理有一个特点:宽高都必须是 2 的倍数。您是知道的,通常影像都是 320 *

   


  应用到D3D中去

  平面的图像、影片看的多了,我们不妨到3D 环境中看看影片。

  我不会在这里介绍 D3D,您要学习它就得自己找资料,这里只是讲在 3D 环境中播放的关键—— 图片到纹理。

  3D 纹理有一个特点:宽高都必须是 2 的倍数。您是知道的,通常影像都是 320 * 240 等大小的,把这个宽高传入创建得到的纹理却是 512 * 256 大小的。所以非得把影像图片拉伸到纹理不可,不然您得到的纹理会有一片黑色,谈不上美观,尽管我的审美能力让我不觉得黄金分割很美丽,但我敢肯定这样的纹理很丑陋。在我的程序中利用了最近点法把图像拉伸为纹理大小。不过这样也引起图像宽高比改变,造成一定程度的扭曲,您可自写个程序把图像保持宽高比拉伸。请看看如何动态改变纹理。

clearcase/" target="_blank" >cccccc width="90%" align=center bgColor=#e7e9e9 border=1>
HRESULT d3d::SetTex(BYTE* pb)
{
 if(!pTexture) return E_FAIL; // 纹理创建不成功
 if(!pb) return E_FAIL; // 指针错误
 // 锁定纹理
 D3DLOCKED_RECT d3dlr;
 if (FAILED(pTexture->LockRect(0, &d3dlr, 0, 0)))
  return E_FAIL;
 BYTE* pTexBits = (BYTE*)d3dlr.pBits; // 取纹理数据区指针
 UINT texPitch = d3dlr.Pitch; // 纹理的 Pitch
 UINT bmpPitch = bmpWid * 4; // 图片的 Pitch
 float xStep = float(bmpWid - 1) / float(texWid - 1); //
 float yStep = float(bmpHei - 1) / float(texHei - 1); //
 BYTE* pNewBits = pTexBits;
 BYTE* pOldBits = pb;
 BYTE* pNewPixel;
 BYTE* pOldPixel;
 // 最近点放大
 for(int y = 0; y < texHei; y ++){
  pOldBits = pb + int(yStep * y) * bmpPitch; // 定位 y
  pNewBits = pTexBits + y * texPitch;
  for(int x = 0; x < texWid; x ++){
   pPixel = pOldBits + 4 * int(xStep * x);// 定位 x
   pNewPixel = pNewBits + 4 * x;
   pNewPixel[0] = pOldPixel[0];
   pNewPixel[1] = pOldPixel[1];
   pNewPixel[2] = pOldPixel[2];
   pNewPixel[3] = 255;// 纹理的 alpha 值,如果启用透明,可更改实现透明效果
  }
 }
 // 解锁纹理
 if (FAILED(pTexture->UnlockRect(0)))
  return E_FAIL;
 return S_OK;
}

  其中 pTexture 是专为影像而设置的纹理,其他不明来历的变量也是 d3d 类的成员,在下面函数中被赋值。

HRESULT d3d::CreateTex(int wid,int hei)
{
 // 根据传入的宽高创建纹理
 if(FAILED(D3DXCreateTexture(this->m_pd3dDevice,wid,hei,1,0,D3DFMT_A8R8G8B8,D3DPOOL_MANAGED,&pTexture))){
  return E_FAIL;
 }
 // 纹理描述
 D3DSURFACE_DESC ddsd;
 if ( FAILED(pTexture->GetLevelDesc( 0, &ddsd ) ) ) {
  return E_FAIL;
 }
 // 核对纹理格式,规定为 A8R8G8B8 的 32bit ARGB 格式
 if(ddsd.Format != D3DFMT_A8R8G8B8){
  pTexture->Release();
  pTexture = NULL;
  return E_FAIL;
 }
 texWid = ddsd.Width; // 纹理宽
 texHei = ddsd.Height;// 纹理高
 bmpWid = wid; // 图片宽
 bmpHei = hei; // 图片高
 return S_OK;
}

  我不想每次使用纹理时创建一个新的,每次渲染后就释放它,这会影响性能,所以在使用中先在恰当的地方,例如按下播放按钮后就创建纹理,然后在有新影像图片到来时更新纹理。我用的是Direct3D8,并非Direct3D9,因为我的 GF4(我很想它消失了,可叹袋中空空如也)运行 D3D9 很慢,简单的场景用 D3D8 也足够了。附加一句,美妙的场景可使影片播放增色不少。

  另外,我也试过在OpenGL中使用影片纹理,不过我的方法使 CPU 占用率达 100%,但有一点可以肯定,这同样可应用于OpenGL。

  敬告:如果您有兴趣读我的程序,请不要试图通过看我的 d3d 类来学习 Direct3D,那只是我前段时间为了学 D3D 而搭的框架类,有很多不规范的地方,而且 3D 物体更是一团糟,所写的代码是临时性的,根本没考虑可读性,现在我见到它们也很头痛,不想修整。所以为了不令您对 D3D 产生不好的印象,也为了保持您对 3D 世界探索的热情,请另行找资料系统学习 D3D。对于我的程序,您只要知道我在哪里使用了 d3d 类的哪些功能来实现效果就行了,切记!

原文转自:http://www.ltesting.net