openGL 纹理使用
最近找了点资料学习了下openGL 纹理的使用
先有个 整体把握, 然后再去看大部头中的细节讲解, 感觉这样的学习方式好些
总结下纹理使用总体流程:
1 启用纹理
glEnable(GL_TEXTURE_2D);
2 载入纹理
3 纹理 的显示
载入纹理
1 读取纹理图像高宽和像素数据到内存中,老版本openGL需要考虑宽度和高度不是的整数次方
2 分配一个新的纹理编号 glGenTextures(1, &texture_ID);
3 绑定纹理 glBindTexture(GL_TEXTURE_2D, texture_ID);表示当前所使用的纹理对象
4 设置纹理参数, 一堆函数调用glTexParameter*
5 载入 glTexImage2D
纹理的显示
显示某个纹理之前先绑定 glBindTexture(GL_TEXTURE_2D, texGround);
指定绘制方法 glBegin(GL_QUADS);
指定纹理坐标和对应顶点 glTexCoord2f(0.0f, 0.0f); glVertex3f(-8.0f, -8.0f, 0.0f);
#define WindowWidth 400
#define WindowHeight 400
#define WindowTitle "OpenGL 纹理测试"
#include <gl/glut.h>
#include <stdio.h>
#include <stdlib.h>
/* 函数 grab
* 抓取窗口中的像素
* 假设窗口宽度为 WindowWidth,高度为 WindowHeight
*/
#define BMP_Header_Length 54
void grab(void)
{FILE* pDummyFile;FILE* pWritingFile;GLubyte* pPixelData;GLubyte BMP_Header[BMP_Header_Length];GLint i, j;GLint PixelDataLength;// 计算像素数据的实际长度i = WindowWidth * 3; // 得到每一行的像素数据长度while( i%4 != 0 ) // 补充数据,直到 i 是的倍数++i; // 本来还有更快的算法,// 但这里仅追求直观,对速度没有太高要求PixelDataLength = i * WindowHeight;// 分配内存和打开文件pPixelData = (GLubyte*)malloc(PixelDataLength);if( pPixelData == 0 )exit(0);pDummyFile = fopen("dummy.bmp", "rb");if( pDummyFile == 0 )exit(0);pWritingFile = fopen("grab.bmp", "wb");if( pWritingFile == 0 )exit(0);// 读取像素glPixelStorei(GL_UNPACK_ALIGNMENT, 4);glReadPixels(0, 0, WindowWidth, WindowHeight,GL_BGR_EXT, GL_UNSIGNED_BYTE, pPixelData);// 把 dummy.bmp 的文件头复制为新文件的文件头fread(BMP_Header, sizeof(BMP_Header), 1, pDummyFile);fwrite(BMP_Header, sizeof(BMP_Header), 1, pWritingFile);fseek(pWritingFile, 0x0012, SEEK_SET);i = WindowWidth;j = WindowHeight;fwrite(&i, sizeof(i), 1, pWritingFile);fwrite(&j, sizeof(j), 1, pWritingFile);// 写入像素数据fseek(pWritingFile, 0, SEEK_END);fwrite(pPixelData, PixelDataLength, 1, pWritingFile);// 释放内存和关闭文件fclose(pDummyFile);fclose(pWritingFile);free(pPixelData);
}/* 函数 power_of_two
* 检查一个整数是否为 2 的整数次方,如果是,返回 1,否则返回 0
* 实际上只要查看其二进制位中有多少个,如果正好有 1 个,返回 1,否则返回 0
* 在“查看其二进制位中有多少个”时使用了一个小技巧
* 使用 n &= (n-1)可以使得 n 中的减少一个(具体原理大家可以自己思考)
*/
int power_of_two(int n)
{if( n <= 0 )return 0;return (n & (n-1)) == 0;
}
/* 函数 load_texture
* 读取一个 BMP 文件作为纹理
* 如果失败,返回 0,如果成功,返回纹理编号
*/
GLuint load_texture(const char* file_name)
{GLint width, height, total_bytes;GLubyte* pixels = 0;GLuint last_texture_ID, texture_ID = 0;// 打开文件,如果失败,返回FILE* pFile = fopen(file_name, "rb");if( pFile == 0 )return 0;// 读取文件中图象的宽度和高度fseek(pFile, 0x0012, SEEK_SET);fread(&width, 4, 1, pFile);fread(&height, 4, 1, pFile);fseek(pFile, BMP_Header_Length, SEEK_SET);// 计算每行像素所占字节数,并根据此数据计算总像素字节数{GLint line_bytes = width * 3;while( line_bytes % 4 != 0 )++line_bytes;total_bytes = line_bytes * height;}// 根据总像素字节数分配内存pixels = (GLubyte*)malloc(total_bytes);if( pixels == 0 ){fclose(pFile);return 0;}// 读取像素数据if( fread(pixels, total_bytes, 1, pFile) <= 0 ){free(pixels);fclose(pFile);return 0;}// 在旧版本的 OpenGL 中// 如果图象的宽度和高度不是的整数次方,则需要进行缩放// 这里并没有检查 OpenGL 版本,出于对版本兼容性的考虑,按旧版本处理// 另外,无论是旧版本还是新版本,// 当图象的宽度和高度超过当前 OpenGL 实现所支持的最大值时,也要进行缩放{GLint max;glGetIntegerv(GL_MAX_TEXTURE_SIZE, &max);if( !power_of_two(width)|| !power_of_two(height)|| width > max|| height > max ){const GLint new_width = 256;const GLint new_height = 256; // 规定缩放后新的大小为边长的正方形GLint new_line_bytes, new_total_bytes;GLubyte* new_pixels = 0;// 计算每行需要的字节数和总字节数new_line_bytes = new_width * 3;while( new_line_bytes % 4 != 0 )++new_line_bytes;new_total_bytes = new_line_bytes * new_height;// 分配内存new_pixels = (GLubyte*)malloc(new_total_bytes);if( new_pixels == 0 ){free(pixels);fclose(pFile);return 0;}// 进行像素缩放gluScaleImage(GL_RGB,width, height, GL_UNSIGNED_BYTE, pixels,new_width, new_height, GL_UNSIGNED_BYTE, new_pixels);// 释放原来的像素数据,把 pixels 指向新的像素数据,并重新设置 width 和 heightfree(pixels);pixels = new_pixels;width = new_width;height = new_height;}}// 分配一个新的纹理编号glGenTextures(1, &texture_ID);if( texture_ID == 0 ){free(pixels);fclose(pFile);return 0;}// 绑定新的纹理,载入纹理并设置纹理参数// 在绑定前,先获得原来绑定的纹理编号,以便在最后进行恢复glGetIntegerv(GL_TEXTURE_BINDING_2D,(GLint *) &last_texture_ID);glBindTexture(GL_TEXTURE_2D, texture_ID);glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);glTexEnvf(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, width, height, 0,GL_BGR_EXT, GL_UNSIGNED_BYTE, pixels);glBindTexture(GL_TEXTURE_2D, last_texture_ID);// 之前为 pixels 分配的内存可在使用 glTexImage2D 以后释放// 因为此时像素数据已经被 OpenGL 另行保存了一份(可能被保存到专门的图形硬件中)free(pixels);return texture_ID;
}/* 两个纹理对象的编号
*/
GLuint texGround;
GLuint texWall;
void display(void)
{// 清除屏幕glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);// 设置视角glMatrixMode(GL_PROJECTION);glLoadIdentity();gluPerspective(75, 1, 1, 21);glMatrixMode(GL_MODELVIEW);glLoadIdentity();gluLookAt(1, 5, 5, 0, 0, 0, 0, 0, 1);// 使用“地”纹理绘制土地// 土地在Z平面上glBindTexture(GL_TEXTURE_2D, texGround);glBegin(GL_QUADS);glTexCoord2f(0.0f, 0.0f); glVertex3f(-8.0f, -8.0f, 0.0f);glTexCoord2f(0.0f, 5.0f); glVertex3f(-8.0f, 8.0f, 0.0f);glTexCoord2f(5.0f, 5.0f); glVertex3f(8.0f, 8.0f, 0.0f);glTexCoord2f(5.0f, 0.0f); glVertex3f(8.0f, -8.0f, 0.0f);glEnd();// 使用“墙”纹理绘制栅栏glBindTexture(GL_TEXTURE_2D, texWall);glBegin(GL_QUADS);glTexCoord2f(0.0f, 0.0f); glVertex3f(-6.0f, -3.0f, 0.0f);glTexCoord2f(0.0f, 1.0f); glVertex3f(-6.0f, -3.0f, 1.5f);glTexCoord2f(5.0f, 1.0f); glVertex3f(6.0f, -3.0f, 1.5f);glTexCoord2f(5.0f, 0.0f); glVertex3f(6.0f, -3.0f, 0.0f);glEnd();// 旋转后再绘制一个glRotatef(-90, 0, 0, 1);glBegin(GL_QUADS);glTexCoord2f(0.0f, 0.0f); glVertex3f(-6.0f, -3.0f, 0.0f);glTexCoord2f(0.0f, 1.0f); glVertex3f(-6.0f, -3.0f, 1.5f);glTexCoord2f(5.0f, 1.0f); glVertex3f(6.0f, -3.0f, 1.5f);glTexCoord2f(5.0f, 0.0f); glVertex3f(6.0f, -3.0f, 0.0f);glEnd();// 交换缓冲区,并保存像素数据到文件glutSwapBuffers();grab();
}
int main(int argc, char* argv[])
{// GLUT 初始化glutInit(&argc, argv);glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGBA);glutInitWindowPosition(100, 100);glutInitWindowSize(WindowWidth, WindowHeight);glutCreateWindow(WindowTitle);glutDisplayFunc(&display);// 在这里做一些初始化glEnable(GL_DEPTH_TEST);glEnable(GL_TEXTURE_2D);texGround = load_texture("ground.bmp");texWall = load_texture("wall.bmp");// 开始显示glutMainLoop();return 0;
}
代码下载
打包下载
openGL 纹理使用相关推荐
- OpenGL 纹理坐标的实例
OpenGL 纹理坐标 先上图,再解答. 完整主要的源代码 源代码剖析 先上图,再解答. 完整主要的源代码 #include <shader.h> #include <vmath.h ...
- OpenGL纹理矩阵,alpha混合和丢弃
OpenGL纹理矩阵,alpha混合和丢弃 先上图,再解答. 完整主要的源代码 源代码剖析 先上图,再解答. 完整主要的源代码 #include <stdio.h> #include &q ...
- OpenGL 纹理Textures
OpenGL纹理 OpenGL纹理简介 纹理环绕方式 纹理过滤 多级渐远纹理 加载与创建纹理 生成纹理 应用纹理 纹理单元 OpenGL纹理简介 我们已经了解到,我们可以为每个顶点添加颜色来增加图形的 ...
- C++ Opengl纹理混合源码
C++ Opengl纹理混合源码 项目开发环境 项目功能 项目演示 项目源码传送门 项目开发环境 开发语言:C++和IDE:VS2017,操作系统Windows版本windows SDK8.1,三方库 ...
- C++ Opengl纹理过滤和光照实例源码
C++ Opengl纹理过滤和光照实例源码 项目开发环境 项目功能 项目演示 项目源码传送门 项目开发环境 开发语言:C++和IDE:VS2017,操作系统Windows版本windows SDK8. ...
- C++ Opengl纹理贴图源码
C++ Opengl纹理贴图源码 项目开发环境 项目功能 项目演示 项目源码传送门 项目开发环境 开发语言:C++和IDE:VS2017,操作系统Windows版本windows SDK8.1,三方库 ...
- android openGl纹理的使用
今天,简单讲讲android关于纹理的知识. 一.纹理的概念 纹理(texture) 在游戏制作里面指贴图,计算机图形学中的纹理既包括通常意义上物体表面的纹理即使物体表面呈现凹凸不平的沟纹,同时也包括 ...
- C++ opengl 纹理生成
程序运行截图如下: 看着很炫酷:其实是图片炫酷 程序源码如下: void Init() {glMatrixMode(GL_PROJECTION);gluPerspective(50.0f, 800.0 ...
- OpenGL纹理本质(三)
几个月前调试3D纹理时发现一个有趣的问题:同样的3D HW lib库在android2.2系统上可以正常工作,但在2.3系统上却不能工作,显示的图像白屏,调试了几天才将问题定位并解决,解决方法很简单: ...
- OpenGL纹理详解
OpenGL纹理详解 现实生活中,纹理最通常的作用是装饰我们的物体模型,它就像是贴纸一样贴在物体表面,使得物体表面拥有图案.但实际上在OpenGL中,纹理的作用不仅限于此,它可以用来存储大量的数据,一 ...
最新文章
- C语言学习之利用指针输出二维数组任一行任一列元素的值
- python初始化方法对应的变量是全局变量嘛_在Python中初始化全局变量的正确方法...
- android开发(49) android 使用 CollapsingToolbarLayout ,可折叠的顶部导航栏
- php cdi_异步CDI事件
- python爬虫网页pdf_爬虫实战【3】Python-如何将html转化为pdf(PdfKit)
- pt-query-digest分析mysql日志
- 代码生成器集合(整理)
- linux 挂载raid_linux下做raid
- YACC介绍(译文)
- 如何使用bat批处理命令打开WSL
- matlab2010a中文,MATLAB 2010a解决中文字体乱码的方法
- Omise携手i2c促进东南亚发卡现代化和加速发卡能力提升
- 接口测试平台代码实现86: 全局请求头-1
- python calu_Python学习笔记4:函数
- 工业相机--海康威视相机
- 计算机专业前端实习生的实习经历
- Git常用命令与Git for windows solarized主题配置
- 五子棋双人对战的实现
- R语言paste()函数的使用
- 知晓云 php,用云函数快速实现图片爬虫