青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

posts - 131, comments - 12, trackbacks - 0, articles - 0
  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

OpenGL可以把紋理映射到指定的圖形的表面上。簡單一點的,就是給平面映射紋理,比如一個四邊形,一個長方體的6個面,都可以指定位圖作為紋理映射到各個面上。

關于將一個位圖作為紋理映射到某個或者多個面上,可以學習Jeff Molofee的OpenGL系列教程。

對于指定的多個紋理,要根據自己的需要映射到不同的面上,需要對位圖創建一個數組,用來存儲位圖的名稱,然后在初始化OpenGL的時候,可以讀取這些位圖,然后生成多個紋理存儲到一個紋理數組中,接著就可以指定繪制的某個面,對該指定的面進行紋理映射。

下面,在的Jeff Molofee教程的第六課的基礎上,實現對6個面分別進行不同的紋理映射。

準備工作就是制作6幅不同的位圖,如圖所示:


關鍵代碼及其說明如下。

創建全局紋理數組

GLuint texture[6];   // 創建一個全局的紋理數組,用來存儲將位圖轉換之后得到的紋理,對應于立方體的6個面

加載位圖文件

加載位圖,也就是把位圖讀取到內存空間,實現紋理的創建,加載位圖的函數說明一下:

AUX_RGBImageRec *LoadBMP(char *Filename)    // 根據位圖文件的名稱進行加載
{
FILE *File=NULL;        // 文件指針

if (!Filename)         // 如果沒有指定位圖文件名稱就返回NULL
{
   return NULL;         
}

File=fopen(Filename,"r");       // 根據指定的位圖文件名稱,打開該位圖文件

if (File)           // 如果位圖文件存在
{
   fclose(File);         // 因為只是需要判斷問題是否存在,而不需要對位圖文件進行寫操作,所以關閉位圖文件
   return auxDIBImageLoad(Filename);   // 其實,只需要一個真正存在的位圖文件的名稱,實現加載位圖文件,并返回一個指針
}

return NULL;          // 位圖文件加載失敗就返回NULL
}

上面實現加載位圖的函數中,AUX_RGBImageRec是glaux.h中定義的類型,該類型的定義如下所示:

/*
** RGB Image Structure
*/

typedef struct _AUX_RGBImageRec {
    GLint sizeX, sizeY;
    unsigned char *data;
} AUX_RGBImageRec;

首先,AUX_RGBImageRec類型是一個RGB圖像結構類型。該結構定義了三個成員:

sizeX —— 圖像的寬度;
sizeY —— 圖像的高度;
data; —— 圖形所包含的數據,其實也就是該圖形在內存中的像素數據的一個指針。

AUX_RGBImageRec類型的變量描述了一幅圖像的特征。

上述函數中,調用了glaux.h庫文件中的auxDIBImageLoad函數,其實它是一個宏,函數原型為auxRGBImageLoadW(LPCWSTR)或者auxRGBImageLoadA(LPCSTR),可以在該庫文件中找到它的定義,如下所示:

/* AUX_RGBImageRec * APIENTRY auxRGBImageLoad(LPCTSTR); */
#ifdef UNICODE
#define auxRGBImageLoad auxRGBImageLoadW
#else
#define auxRGBImageLoad auxRGBImageLoadA
#endif
AUX_RGBImageRec * APIENTRY auxRGBImageLoadA(LPCSTR);
AUX_RGBImageRec * APIENTRY auxRGBImageLoadW(LPCWSTR);

#ifdef UNICODE
#define auxDIBImageLoad auxDIBImageLoadW
#else
#define auxDIBImageLoad auxDIBImageLoadA
#endif
AUX_RGBImageRec * APIENTRY auxDIBImageLoadA(LPCSTR);
AUX_RGBImageRec * APIENTRY auxDIBImageLoadW(LPCWSTR);

宏auxDIBImageLoad實現的功能就是:根據指定的位圖名稱,將該位圖的信息加載到內存中,以便用來創建成為紋理。

創建紋理并加載紋理

用于創建并加載紋理的函數為LoadGLTextures,實現如下所示:

int LoadGLTextures()         // 根據加載的位圖創建紋理
{
int Status=FALSE;         // 指示紋理創建是否成功的標志

AUX_RGBImageRec *TextureImage[6];     // 創建一個紋理圖像數組,這里指定數組大小為6

memset(TextureImage,0,sizeof(void *)*6);          // 初始化紋理圖像數組,為其分配內存

char *pictures[] = {// 創建一個位圖名稱數組,對應6幅位圖
   "Data/No1.bmp",
   "Data/No2.bmp",
   "Data/No3.bmp",
   "Data/No4.bmp",
   "Data/No5.bmp",
   "Data/No6.bmp"
};
for(int i=0; i<6; i++)// 遍歷位圖名稱數組,根據位圖名稱分別生成
{
   if (TextureImage[i]=LoadBMP(pictures[i]))// 加載位圖i成功,修改狀態標志變量Status為TRUE
   {
    Status=TRUE;         

   glGenTextures(1, &texture[i]);     // 為第i個位圖創建紋理
   glBindTexture(GL_TEXTURE_2D, texture[i]);// 將生成的紋理的名稱綁定到指定的紋理上
   glTexImage2D(GL_TEXTURE_2D, 0, 3, TextureImage[i]->sizeX, TextureImage[i]->sizeY, 0, GL_RGB, GL_UNSIGNED_BYTE, TextureImage[i]->data);
   glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
   }

   if (TextureImage[i])         // 釋放位圖數組占用的內存空間
   {
    if (TextureImage[i]->data)       
    {
     free(TextureImage[i]->data);    
    }

    free(TextureImage[i]);        
   }
}
return Status;          // 創建紋理并加載,返回成功或者失敗的標志Status
}

上述函數是創建和加載紋理的核心實現。

1、glGenTextures函數

其中,調用了glGenTextures函數,查看MSDN可以看到,聲明如下所示:

void glGenTextures(
GLsizein,         
GLuint *textures  
);

函數參數的含義:

n—— 生成的紋理的名稱的個數;

textures—— 生成的紋理名稱所存儲位置的指針,也就是一個紋理數組的內存地址,或者說是數組首元素的內存地址。

函數被調用,會生成一系列紋理的名字,并存儲到指定的數組中。

2、glBindTexture函數

glBindTexture函數實現了將調用glGenTextures函數生成的紋理的名字綁定到對應的目標紋理上。該函數的聲明如下所示:

void glBindTexture(
GLenumtarget,  
GLuinttexture  
);

函數參數的含義:

target—— 紋理被綁定的目標,它只能取值GL_TEXTURE_1D或者GL_TEXTURE_2D;

texture—— 紋理的名稱,并且,該紋理的名稱在當前的應用中不能被再次使用。

3、glTexImage2D函數

調用glTexImage2D函數,用來指定二維紋理圖像。該函數的聲明如下所示:

void glTexImage2D(
GLenumtarget,       
GLintlevel,         
GLintcomponents,    
GLsizeiwidth,       
GLsizeiheight,      
GLintborder,        
GLenumformat,       
GLenumtype,         
const GLvoid*pixels
);

函數參數的含義:

target—— 指定目標紋理,必須為GL_TEXTURE_2D;

level—— 指定圖像級別的編號,0表示基本圖像,其它可以參考MSDN;

components—— 紋理中顏色組件的編號,可是是1或2或3或4;

width—— 紋理圖像的寬度;

height—— 紋理圖像的高度;

border—— 紋理圖像的邊框寬度,必須是0或1;

format—— 指定像素數據的格式,一共有9個取值:GL_COLOR_INDEX、GL_RED、GL_GREEN、GL_BLUE、GL_ALPHA、GL_RGB、GL_RGBA、GL_BGR_EXT、GL_BGRA_EXT、GL_LUMINANCE、GL_LUMINANCE_ALPHA ,具體含義可以參考MSDN;

type—— 像素數據的數據類型,取值可以為GL_UNSIGNED_BYTE, GL_BYTE, GL_BITMAP, GL_UNSIGNED_SHORT, GL_SHORT, GL_UNSIGNED_INT, GL_INT, and GL_FLOAT;

pixels—— 內存中像素數據的指針。

4、glTexParameteri函數

glTexParameteri函數或者glTexParameterf函數用來設置紋理參數,聲明如下所示:

void glTexParameterf(
GLenumtarget,
GLenumpname,
GLfloatparam
);

void glTexParameteri(
GLenumtarget,
GLenumpname,
GLintparam   
);

函數參數的含義:

target—— 目標紋理,必須為GL_TEXTURE_1D或GL_TEXTURE_2D;

pname—— 用來設置紋理映射過程中像素映射的問題等,取值可以為:GL_TEXTURE_MIN_FILTER、GL_TEXTURE_MAG_FILTER、GL_TEXTURE_WRAP_S、GL_TEXTURE_WRAP_T,詳細含義可以查看MSDN;

param—— 實際上就是pname的值,可以參考MSDN。

另外,該類函數還有兩個:

void glTexParameterfv(
GLenumtarget,        
GLenumpname,         
const GLfloat*params
);

void glTexParameteriv(
GLenumtarget,      
GLenumpname,       
const GLint*params
);

上述程序中調用如下:

    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);

功能就是實現線形濾波的功能,當紋理映射到圖形表面以后,如果因為其它條件的設置導致紋理不能更好地顯示的時候,進行過濾,按照指定的方式進行顯示,可能會過濾掉顯示不正常的紋理像素。

紋理映射過程

紋理映射的過程是在DrawGLScene函數中實現的,也就是在繪制圖形的過程中,直接進行我呢里映射,或者稱為,為指定的平面貼紋理,DrawGLScene函數實現如下所示:

int DrawGLScene(GLvoid)         
{
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
glLoadIdentity();         
glTranslatef(0.0f,0.0f,-5.0f);

glRotatef(xrot,1.0f,0.0f,0.0f);
glRotatef(yrot,0.0f,1.0f,0.0f);
glRotatef(zrot,0.0f,0.0f,1.0f);


  // Front Face
glBindTexture(GL_TEXTURE_2D, texture[0]);//   選擇第一個紋理texture[0],進行貼紋理
glBegin(GL_QUADS);
  glTexCoord2f(0.0f, 0.0f); glVertex3f(-1.0f, -1.0f, 1.0f);
   glTexCoord2f(1.0f, 0.0f); glVertex3f( 1.0f, -1.0f, 1.0f);
   glTexCoord2f(1.0f, 1.0f); glVertex3f( 1.0f, 1.0f, 1.0f);
   glTexCoord2f(0.0f, 1.0f); glVertex3f(-1.0f, 1.0f, 1.0f);
glEnd();

  // Back Face
glBindTexture(GL_TEXTURE_2D, texture[1]);//   選擇第二個紋理texture[1],進行貼紋理
glBegin(GL_QUADS);
   glTexCoord2f(1.0f, 0.0f); glVertex3f(-1.0f, -1.0f, -1.0f);
   glTexCoord2f(1.0f, 1.0f); glVertex3f(-1.0f, 1.0f, -1.0f);
   glTexCoord2f(0.0f, 1.0f); glVertex3f( 1.0f, 1.0f, -1.0f);
   glTexCoord2f(0.0f, 0.0f); glVertex3f( 1.0f, -1.0f, -1.0f);
glEnd();

  // Top Face
glBindTexture(GL_TEXTURE_2D, texture[2]);//   選擇第三個紋理texture[2],進行貼紋理
glBegin(GL_QUADS);
   glTexCoord2f(0.0f, 1.0f); glVertex3f(-1.0f, 1.0f, -1.0f);
   glTexCoord2f(0.0f, 0.0f); glVertex3f(-1.0f, 1.0f, 1.0f);
   glTexCoord2f(1.0f, 0.0f); glVertex3f( 1.0f, 1.0f, 1.0f);
   glTexCoord2f(1.0f, 1.0f); glVertex3f( 1.0f, 1.0f, -1.0f);
glEnd();

  // Bottom Face
glBindTexture(GL_TEXTURE_2D, texture[3]);//   選擇第四個紋理texture[3],進行貼紋理
glBegin(GL_QUADS);
   glTexCoord2f(1.0f, 1.0f); glVertex3f(-1.0f, -1.0f, -1.0f);
   glTexCoord2f(0.0f, 1.0f); glVertex3f( 1.0f, -1.0f, -1.0f);
   glTexCoord2f(0.0f, 0.0f); glVertex3f( 1.0f, -1.0f, 1.0f);
   glTexCoord2f(1.0f, 0.0f); glVertex3f(-1.0f, -1.0f, 1.0f);
glEnd();

  // Right face
glBindTexture(GL_TEXTURE_2D, texture[4]);//   選擇第五個紋理texture[4],進行貼紋理
glBegin(GL_QUADS);
   glTexCoord2f(1.0f, 0.0f); glVertex3f( 1.0f, -1.0f, -1.0f);
   glTexCoord2f(1.0f, 1.0f); glVertex3f( 1.0f, 1.0f, -1.0f);
   glTexCoord2f(0.0f, 1.0f); glVertex3f( 1.0f, 1.0f, 1.0f);
   glTexCoord2f(0.0f, 0.0f); glVertex3f( 1.0f, -1.0f, 1.0f);
glEnd();

  // Left Face
glBindTexture(GL_TEXTURE_2D, texture[5]);//   選擇第六個紋理texture[5],進行貼紋理
glBegin(GL_QUADS);
   glTexCoord2f(0.0f, 0.0f); glVertex3f(-1.0f, -1.0f, -1.0f);
   glTexCoord2f(1.0f, 0.0f); glVertex3f(-1.0f, -1.0f, 1.0f);
   glTexCoord2f(1.0f, 1.0f); glVertex3f(-1.0f, 1.0f, 1.0f);
   glTexCoord2f(0.0f, 1.0f); glVertex3f(-1.0f, 1.0f, -1.0f);
glEnd();

xrot+=0.3f;
yrot+=0.2f;
zrot+=0.4f;
return TRUE;          
}

因為,通過前面的過程,已經將位圖加載并創建和加載紋理成功,紋理數組已經存在于內存之中,調用上述函數實現紋理映射,即,從內存中取出指定的紋理,將其映射到立方體的指定的面上。

上述函數中調用了glTexCoord2f函數,設置紋理坐標,該函數的聲明如下所示:

void glTexCoord2f(
GLfloats,
GLfloatt
);

glTexCoord2f 的第一個參數是X坐標,當s=0.0f 時是紋理的左側,s=0.5f 時是紋理的中點,s=1.0f 時是紋理的右側。 glTexCoord2f 的第二個參數是Y坐標,t=0.0f 是紋理的底部,t=0.5f 是紋理的中點, t=1.0f 是紋理的頂部。

上述函數在為前面那個面映射紋理的時候調用如下:

   // Front Face
glBindTexture(GL_TEXTURE_2D, texture[0]);
glBegin(GL_QUADS);
   glTexCoord2f(0.0f, 0.0f); glVertex3f(-1.0f, -1.0f, 1.0f);
   glTexCoord2f(1.0f, 0.0f); glVertex3f( 1.0f, -1.0f, 1.0f);
   glTexCoord2f(1.0f, 1.0f); glVertex3f( 1.0f, 1.0f, 1.0f);
   glTexCoord2f(0.0f, 1.0f); glVertex3f(-1.0f, 1.0f, 1.0f);
glEnd();

其中:

glTexCoord2f(0.0f, 0.0f);表示將紋理texture[0]的左下角坐標(0.0f, 0.0f)映射到立方體前面那個面的頂點(-1.0f, -1.0f, 1.0)上;

glTexCoord2f(1.0f, 0.0f);表示將紋理texture[0]的右下角坐標(1.0f, 0.0f)映射到立方體前面那個面的頂點(1.0f, -1.0f, 1.0f)上;

glTexCoord2f(1.0f, 1.0f);表示將紋理texture[0]的右上角坐標(1.0f, 1.0f)映射到立方體前面那個面的頂點(1.0f, 1.0f, 1.0f)上;

glTexCoord2f(0.0f, 1.0f);表示將紋理texture[0]的左上角坐標(0.0f, 1.0f)映射到立方體前面那個面的頂點(-1.0f, 1.0f, 1.0f)上。

這樣,紋理texture[0]就被映射到了立方體前面那個面上。

紋理映射結果

為了使立方體能夠運動起來,對立方體進行累的旋轉變換,而且,定義了三個全局變量:

GLfloat xrot;    // 沿著x旋轉的變量
GLfloat yrot;    // 沿著y旋轉的變量
GLfloat zrot;   // 沿著z旋轉的變量

初始化都為0,然后,在每次調用DrawGLScene函數的時候,改變x、y、z的分量值,在DrawGLScene函數中如下所示:

xrot+=0.3f;
yrot+=0.2f;
zrot+=0.4f;

在DrawGLScene函數中還要執行旋轉變換:

glRotatef(xrot,1.0f,0.0f,0.0f);
glRotatef(yrot,0.0f,1.0f,0.0f);
glRotatef(zrot,0.0f,0.0f,1.0f);

每次重繪都在改變旋轉軸,所以我們繪制的立方體能夠動起來,看到各個進行紋理映射的面的效果,如圖所示





只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久福利视频导航| 1000部精品久久久久久久久| 激情小说亚洲一区| 亚洲精品网站在线播放gif| 在线国产精品播放| 亚洲男同1069视频| 亚洲视屏一区| 欧美日韩免费视频| 欧美国产丝袜视频| 亚洲成人在线网| 欧美一级欧美一级在线播放| 亚洲女人天堂av| 欧美日韩高清在线播放| 欧美国产第二页| 伊人久久亚洲美女图片| 久久精品理论片| 久久精品视频99| 国产女同一区二区| 亚洲欧美成人一区二区在线电影| 亚洲最新视频在线播放| 欧美精品成人在线| 亚洲卡通欧美制服中文| 99精品欧美一区| 欧美精品激情在线观看| 亚洲美女一区| 在线视频一区观看| 欧美色网在线| 亚洲一级在线| 久久成人精品一区二区三区| 国产麻豆综合| 欧美在线播放视频| 欧美91大片| 亚洲免费成人av电影| 欧美日韩国产在线播放| 一区二区三区欧美亚洲| 在线观看91精品国产麻豆| 久久久国产成人精品| 美女主播精品视频一二三四| 亚洲人成毛片在线播放女女| 欧美精品日韩精品| 亚洲欧美国产三级| 久久综合久久美利坚合众国| 在线观看亚洲a| 欧美激情精品久久久久久蜜臀| 91久久精品美女| 亚洲视频你懂的| 国产一区二区三区奇米久涩| 久久这里只精品最新地址| 亚洲高清一区二区三区| 亚洲一区二区三区精品视频| 国产精品日韩久久久| 久久久国产91| 亚洲精品中文在线| 久久激情一区| 日韩视频免费观看| 国产精品系列在线播放| 裸体素人女欧美日韩| 一本一本久久| 美女日韩在线中文字幕| 一区二区三区精品国产| 国产一二三精品| 欧美日韩国产精品一区二区亚洲| 午夜精彩国产免费不卡不顿大片| 亚洲第一在线综合网站| 欧美一区二区视频在线观看| 亚洲国产另类久久久精品极度| 欧美小视频在线| 免费亚洲一区二区| 午夜精品久久久久久久| 亚洲精品在线三区| 免费不卡在线视频| 欧美一区91| 日韩视频在线观看| 狠狠网亚洲精品| 国产精品黄视频| 欧美国产一区二区| 久久九九热免费视频| 国产精品99久久久久久有的能看| 欧美激情一区二区三区在线| 久久久www成人免费毛片麻豆| 夜久久久久久| 亚洲日韩欧美视频一区| 激情综合色丁香一区二区| 国产精品看片你懂得| 欧美精品成人| 欧美福利一区| 欧美jizz19hd性欧美| 久久精品亚洲一区二区| 亚洲欧美日本视频在线观看| 亚洲精品系列| 最新日韩欧美| 欧美激情精品久久久久久| 久久综合久久综合久久| 欧美在线视频免费播放| 亚洲午夜小视频| 一区二区三区视频在线观看| 亚洲巨乳在线| 亚洲美女在线一区| 99精品国产在热久久下载| 亚洲欧洲在线看| 亚洲肉体裸体xxxx137| 亚洲国产高清一区| 亚洲国产精品久久久久秋霞不卡 | 国产伦理一区| 国产精品狼人久久影院观看方式| 欧美日韩国产精品一区| 欧美日韩精品高清| 欧美日韩日日骚| 国产精品mm| 国产精品男女猛烈高潮激情| 国产精品毛片一区二区三区| 国产精品久久久久99| 国产精品国产a| 国产麻豆精品theporn| 国产毛片久久| 一区福利视频| 亚洲精品美女在线| 一区二区欧美在线观看| 亚洲制服av| 久久国产福利国产秒拍| 久久亚洲精品视频| 亚洲高清成人| 999在线观看精品免费不卡网站| 99在线视频精品| 欧美亚洲综合网| 久久蜜桃av一区精品变态类天堂| 欧美aa在线视频| 国产精品福利av| 国内精品免费在线观看| 91久久在线播放| 亚洲欧美日韩精品一区二区| 久久久人人人| 亚洲国产经典视频| 这里只有精品视频在线| 久久精品免费| 欧美日韩视频不卡| 国内自拍一区| 夜夜嗨av色综合久久久综合网| 亚洲欧美在线观看| 欧美福利视频在线| 中国成人在线视频| 久久天天综合| 国产精品久久中文| 亚洲欧洲日产国码二区| 午夜精品福利一区二区三区av| 欧美91福利在线观看| 一区二区三区精品视频| 久久久久这里只有精品| 欧美日韩精品免费观看视一区二区 | 韩国女主播一区二区三区| 91久久精品日日躁夜夜躁欧美| 亚洲一区二区三区欧美| 美女主播精品视频一二三四| 中日韩视频在线观看| 免费高清在线视频一区·| 国产精品伦一区| 亚洲精品一区二区三区在线观看| 欧美影院成人| 亚洲美女黄色| 久久一区二区三区四区| 国产噜噜噜噜噜久久久久久久久| 亚洲精品一区二区三区四区高清| 久久福利精品| 一区二区三区精品视频| 欧美大秀在线观看| 樱花yy私人影院亚洲| 午夜精品国产更新| 亚洲精品国产系列| 久久久免费av| 国产午夜精品福利| 亚洲一区二区av电影| 亚洲韩国精品一区| 久久综合狠狠综合久久综合88| 国产欧美在线| 午夜精彩国产免费不卡不顿大片| 亚洲人体1000| 欧美91大片| 亚洲国产精品福利| 久久一本综合频道| 欧美一区国产在线| 国产精品制服诱惑| 亚洲欧美成人网| 中国女人久久久| 欧美色图一区二区三区| aⅴ色国产欧美| 亚洲区一区二| 欧美日韩国产大片| 一本一道久久综合狠狠老精东影业 | 欧美性片在线观看| 亚洲永久在线| 亚洲天堂av在线免费观看| 国产精品v片在线观看不卡 | 99亚洲视频| 亚洲激情在线观看| 欧美精品激情在线观看| 一本色道久久综合亚洲精品按摩| 亚洲国产高清一区| 你懂的成人av| 一道本一区二区| 一区二区三区欧美亚洲|