用于为使用固定功能的多纹理单元提供附加的操作功能。包括:不同的纹理坐标,颜色操作,alpha操作,凹凸映射/环境映射。
--------------------------------------------
2,纹理坐标
对于一个顶点,最多允许使用八组纹理坐标(D3DFVF_TEX1到D3DFVF_TEX8),可以利用D3DTSS_TEXCOORDINDEX来选择某组特定的纹理坐标:
m_pd3dDevice->SetTexture(0,m_pWallTexture);上边第一张纹理使用了顶点的第一组纹理坐标,第二张纹理使用了顶点的第二组纹理坐标。当然也可以让第二张纹理也使用第一组纹理坐标,这要是情况而定。
m_pd3dDevice->SetTextureStateState(0,D3DTSS_TEXCOORDINDEX,0);
m_pd3dDevice->SetTexture(1,m_pWallTexture);
m_pd3dDevice->SetTextureStateState(1,D3DTSS_TEXCOORDINDEX,1);
--------------------------------------------
3,四种纹理寻址模式
纹理寻址方式实在是我又查询了其他一些网站之后才知道的事情,因为以前用openGL的时候对纹理映射方式一点都没有接触过,因此这部分看起来就的确显得有点难度了。
一般来说,分配的U、V纹理坐标值都在0.0到1.0范围内(包括它们)。但是,如果我们分配了超出这个范围的纹理坐标,可能会得到一些特别的纹理效果。通过设置纹理寻址模式,我们就可以在纹理坐标超出范围时进行控制,纹理寻址模式就是用来干这个的。
包装(Wrapping)纹理寻址模式
一次镜像寻址模式
这个模式由D3DTEXTUREADDRESS枚举类型的D3DTADDRESS_MIRRORONCE成员确定,是从DirectX8才引入的新的纹理寻址模式,类似于镜像模式于夹持模式的组合。纹理再-1.0到1.0之间做镜像,而在该范围之外做夹持。
注意,上图中A、B两点的最短连线穿过了纹理的中间部分。U、V纹理Wrapping的使用会影响Direct3D在U、V方向上对纹理坐标间最短连线的选取。现在我们假定0.0与1.0重合,那么通过定义,纹理Wrapping就会导致光栅在纹理坐标设置之间来选择最短距离。我们可以认为一个方向上的纹理Wrapping就是让系统认为将一个纹理包裹在了一个圆筒上,就象下图中那样:
上图中我们在U方向上进行了Wrapping,它影响了系统对纹理坐标进行的插值操作。我们使用同样的两个点A和B,可以看到,它们之间最短的连线不再通过纹理的中间部分;它现在穿越了0.0和1.0所在的交界线。沿V方向的Wrapping与它相似,只不过纹理所包裹的圆筒横躺在地上。U、V方向上同时进行Wrapping比较复杂,这时我们可以将纹理想象成一个园环面或者是面包圈的形状。
5,纹理过滤和抗锯齿(反走样Anti-Aliasing)
纹理过滤和反走样解决的是同一类问题,就是屏幕锯齿的问题,不同的是文理过滤解决的是纹理映射过程中的锯齿,而反走样解决的是几何体边沿的锯齿。
一个mipmap就是一系列的纹理,每一幅纹理都与前一幅是相同的图样,但是分辨率都要比前一幅有所降低。mipmap中的每一幅或者每一级图象的高和宽都比前一级小二分之一。Mipmap并不一定必须是正方形。
高分辨率的mipmap图象用于接近观察者的物体。当物体逐渐远离观察者时,使用低分辨率的图象。Mipmap可以提高场景渲染的质量,但是它的内存消耗却很大。
Direct3D将mipmap描绘成一系列相互联系的表面。高分辨率的纹理位于开始处,并与下一级纹理相互联系。以此类推,纹理相互联系,逐渐排列到分辨率最小的一级。
下面这套插图显示了这样的一个例子。这套纹理是一个三维场景中一个集装箱的标签。当我们创建了一个mipmap时,分辨率最高的一幅纹理就是这一套纹理的第一个。这套mipmap中的每一个纹理宽高都是前一个纹理宽高的二分之一。这样,最大分辨率的纹理是256x256,接下来的纹理就是128x128,最后一个纹理就是64x64。
我们有一个能看到这个标签的最大距离。如果观察者从远处向标签走近,那么场景中首先会显示最小的一幅纹理,它的大小是64x64的。
当观察者走进标签时,我们就使用更高分辨率一幅纹理;当观察者走到允许的最近距离时,我们使用分辨率最高的那幅纹理.
这是方法能够模拟纹理的透视效果并能够减少处理时的计算量。与将一幅纹理用于不同的分辨率相比,这种方法更加快速。
过滤是指通过给定的UV坐标从纹理贴图中获取图素的一种方法。一张64*64的纹理映射到400*400象素的多边形就会出现因放大而造成的锯齿,这称为放大问题;一张64*64的纹理映射到10*10象素的多边形的时候,一个屏幕象素对应多个纹理图素,就会因交替占有象素而出现抖动。那么为了解决这个问题,就要采用纹理过滤方法。在默认的显卡设置中,纹理过滤的方法是由驱动程序控制的。
最近点采样是最简单的过滤方法,顾名思义,就是采用离象素最近的图素作为颜色。将纹理坐标对齐到最接近的整数,再将那个位于整数坐标的纹理图素作为最终的颜色。这种方法在处理图素边界的时候会发生一些错误。
线性纹理过滤在Direct3D中采用的是双线性纹理过滤,它计算相对于采样点距离最近的4个图素的平均值。采用如下形式进行设置:
m_pd3Device->SetSamplerState(0,D3DSAMP_MINFILTER,D3DTEXF_LINEAR);
m_pd3Device->SetSamplerState(0,D3DSAMP_MAGFILTER,D3DTEXF_LINEAR);
三线性过滤:选择两张最接近的mipmap,将它们双线性过滤为两张理想大小的mipmap,然后根据理想的mip级组合这两张过滤厚的mipmap中的对应象素。采用如下形式进行设置:
m_pd3Device->SetSamplerState(0,D3DSAMP_MINFILTER,D3DTEXF_LINEAR);
m_pd3Device->SetSamplerState(0,D3DSAMP_MAGFILTER,D3DTEXF_LINEAR);
m_pd3Device->SetSamplerState(0,D3DSAMP_MIPFILTER,D3DTEXF_LINEAR);
各向异性过滤:双线性过滤和三线性过滤的缺点是他们都使用正方形采样区域来采样图素,如果纹理是侧着朝向视点的化则会发生一种称之为“各向异性”的失真效果。那么各向异性过滤则会把因为透视投影拉伸的几何体大小映射回纹理空间中,使得纹理在投射时候被伸张,因此来获得更佳的深度细节和精确显示。使用各向异性过滤的代码如下:
m_pd3Device->SetSamplerState(0,D3DSAMP_MINFILTER,D3DTEXF_ANISOTROPIC);
m_pd3Device->SetSamplerState(0,D3DSAMP_MAGFILTER,D3DTEXF_ANISOTROPIC);
m_pd3Device->SetSamplerState(0,D3DSAMP_MAXD3DTEXF_ANISOTROOPY,16);
全屏反走样(FSAA):采用全屏多采样对每个象素进行对此采样,进行混合后进行输出,通过这种方式来调整图像中每条斜线周围的亮度来隐藏骑上的锯齿效果,它沿着这些边沿产生局部模糊的效果。一种更为高级的多采样成为“可屏蔽多采样(maskable multisampling)”。
alpha混合不知道怎么搞到这一章来介绍,可能是为了利用这一个没有程序的篇章把该介绍的基本知识介绍完吧。我也姑且写到这里。
alpha混合的原理很简单,就是利用混合因子把要绘制到帧缓存中的颜色和当前帧缓存中的颜色进行一个叠加,因此可以实现多次渲染,从而实现许多特效。对应的代码像下面这种形式:
m_pd3dDevice->SetRenderState(D3DRS_ALPHABLENDENABLE,TRUE);
m_pd3dDevice->SetRenderState(D3DRS_SRCBLEND,D3DBLEND_ZERO);
m_pd3dDevice->SetRenderState(D3DRS_DESTBLEND,D3DBLEND_SRCCOLOR);
“Wrapping”纹理寻址模式由D3DTEXTUREADDRESS枚举类型的D3DTADDRESS_WRAP成员来确定,它使Direct3D在每一个整数结点(integer junction)对纹理进行重复。假设我们要创建一个正方形图元,并将纹理坐标声明为(0.0,0.0)、(0.0,3.0)、(3.0,3.0)和(3.0,0.0)。这时,如果我们设置了纹理寻址模式,就可以使纹理在U、V方向都重复三次。,如下图所示:
这种纹理寻址模式的效果与“镜像”模式比较相似,但在本质上是不同的。
“镜像”纹理寻址模式由D3DTEXTUREADDRESS枚举类型的D3DTADDRESS_MIRROR成员来确定,它使Direct3D在每个整数边界处(integer boundary)对纹理进行镜像处理。想在我们创建一个正方形图元,为坐标为(0.0,0.0)、(0.0,3.0)、(3.0,3.0)和(3.0,0.0)。我们设置镜像纹理寻址模式,纹理在U、V方向都重复了三次,并且每一行、每一列都与相邻的行和列成镜像关系。如下图所示:
“钳位”纹理寻址模式由D3DTEXTUREADDRESS枚举类型的D3DTADDRESS_CLAMP成员确定,它使Direct3D将纹理坐标钳制在[0.0, 1.0]范围内。也就是说,它只使用一次纹理,然后将边缘像素的颜色抹去。我们创建一个正方形图元,纹理地址分配为(0.0,0.0)、(0.0,3.0)、(3.0,3.0)和(3.0,0.0)。这时,设置钳位纹理寻址模式,纹理将只使用一次,并且最顶一行和最后一列上的像素颜色会一直延伸到图元的最顶端和最右段,如下图所示:
“边缘颜色”纹理寻址模式由D3DTEXTUREADDRESS枚举类型的D3DTADDRESS_BORDER成员确定,它使Direct3D可以在纹理坐标超过范围的地方使用一个任意的颜色,也就是边界颜色。
下图中展示了一个使用了纹理的图元,它使用了红色的边界色: