首先先从图像中读取纹理:
Program part1:
void SetUpTextureMaps( void ) { // This texture object is for the ceiling texture map. GLuint ceilingTexObj; const char ceilingTexFile[] = "images/ceiling.jpg"; glGenTextures( 1, &ceilingTexObj ); glBindTexture( GL_TEXTURE_2D, ceilingTexObj ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR ); if ( ReadImageFile( ceilingTexFile, &imageData, &imageWidth, &imageHeight, &numComponents ) == 0 ) exit( 1 ); if ( numComponents != 3 ) { fprintf( stderr, "Error: Texture image is not in RGB format.\n" ); exit( 1 ); } gluBuild2DMipmaps( GL_TEXTURE_2D, GL_RGB, imageWidth, imageHeight, GL_RGB, GL_UNSIGNED_BYTE, imageData ); DeallocateImageData( &imageData ); }
从代码中可以看到, glGenTextures( 1, &ceilingTexObj )为当前从文件中读入的纹理分配了一个索引值(其实就是一个整数),将其设置为当前active的纹理,然后从文件中读入纹理,这样程序就知道该所以对应的纹理就是读入的这个;
接着,使用纹理,用纹理来进行纹理贴图:
Program part2:
void DrawRoom( void ){ const float ROOM_HALF_WIDTH = ROOM_WIDTH / 2.0f; glTexEnvf( GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE );// Ceiling. GLfloat matAmbient1[] = { 0.7, 0.7, 0.7, 1.0 }; GLfloat matDiffuse1[] = { 0.7, 0.7, 0.7, 1.0 }; GLfloat matSpecular1[] = { 0.3, 0.3, 0.3, 1.0 }; GLfloat matShininess1[] = { 32.0 }; glMaterialfv( GL_FRONT_AND_BACK, GL_AMBIENT, matAmbient1 ); glMaterialfv( GL_FRONT_AND_BACK, GL_DIFFUSE, matDiffuse1 ); glMaterialfv( GL_FRONT_AND_BACK, GL_SPECULAR, matSpecular1 ); glMaterialfv( GL_FRONT_AND_BACK, GL_SHININESS, matShininess1 ); glBindTexture( GL_TEXTURE_2D, ceilingTexObj ); glNormal3f( 0.0, -1.0, 0.0 ); // Normal vector. SubdivideAndDrawQuad( 24, 24, 0.0, 0.0, ROOM_HALF_WIDTH, ROOM_HEIGHT, ROOM_HALF_WIDTH, ROOM_WIDTH, 0.0, -ROOM_HALF_WIDTH, ROOM_HEIGHT, ROOM_HALF_WIDTH, ROOM_WIDTH, ROOM_WIDTH, -ROOM_HALF_WIDTH, ROOM_HEIGHT, -ROOM_HALF_WIDTH, 0.0, ROOM_WIDTH, ROOM_HALF_WIDTH, ROOM_HEIGHT, -ROOM_HALF_WIDTH );}
在这里我们使用了ceilingTexObj ,它指向第一部分程序读入的texture,将其设为当前active的纹理,然后实现贴图。
备注:SubdivideAndDrawQuad()实现了纹理坐标与面坐标的对应。
Extra information
当要求用我们自己从某一视点看到的场景来作为纹理的话,我们应该设置好视点的位置,然后进行view,这时我们看到的内容存在后缓冲区里,最后从后缓冲区中读取纹理,进行贴图。
这里我们主要要做的是对比使用buffer中的纹理与使用从图像中读取的纹理在设置纹理时的区别(使用函数SetUpTextureMaps()进行纹理设置):
Program part3:
void SetUpTextureMaps( void ){// This texture object is for storing the reflection image. glGenTextures( 1, &reflectionTexObj ); glBindTexture( GL_TEXTURE_2D, reflectionTexObj ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR ); glTexParameteri( GL_TEXTURE_2D, GL_GENERATE_MIPMAP, GL_TRUE );}
对比program part1与program part3,区别在于,part1读入图片之后使用gluBuild2DMipmaps( GL_TEXTURE_2D, GL_RGB, imageWidth, imageHeight, GL_RGB, GL_UNSIGNED_BYTE, imageData ) 来生成Mipmaps纹理即用imageData来生成Mipmap;而在part3中我们用glTexParameteri( GL_TEXTURE_2D, GL_GENERATE_MIPMAP, GL_TRUE ),即设置将GL_GENERATE_MIPMAP设置为GL_TRUE即可,就能从后缓存中的场景产生Mipmap.
Draw时,只需激活即glBindTexture,绘制的时候,程序就会运用该纹理的索引reflectionTexObj 指向内存中该图像,或者我们view后场景在内存中的位置。
// Read color buffer into texture object.
glReadBuffer( GL_BACK ); glBindTexture( GL_TEXTURE_2D, reflectionTexObj ); glCopyTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, 0, 0, winWidth, winHeight, 0);这一步完成了从back buffer中读取我们自己view的场景作为纹理。