OpenGL-ES 学习(15) ----纹理

目录

      • 纹理简介
      • 纹理映射
      • 纹理映射流程
      • 示例代码:
      • 纹理的环绕和过滤方式
      • 纹理的过滤方式

纹理简介

现实生活中,纹理(Texture) 类似于游戏中皮肤的概念,最通常的作用是装饰 3D 物体,它像贴纸一样贴在物体的表面,丰富物体的表面和细节
OpenGL-ES 开发中,纹理除了用于装饰物体表面,还可以用来作为存储数据的容器
所以在 OpenGL-ES 中,纹理实际上是一个可以被采样的复杂数据集合,是 GPU 的图像数据结构,纹理分为 2D纹理、立方图纹理和 3D纹理

  • 2D 纹理是 OpenGL-ES 中最常见和最常用的纹理形式,是一个表示图像数据的二维数组,纹理中一个单独的数据单元被称为纹素或者纹理像素
  • 立方图纹理(CubeMap)是一个由6个单独的 2D 纹理面组成的纹理,立方图纹理像素的读取是使用三维坐标(s, t, r)作为纹理坐标
  • 3D 纹理 可以看做 2D 纹理的集合,2D 纹理是 3D 纹理的一个切面,使用三维坐标对齐进行访问

纹理映射

OpenGL-ES 中,纹理映射就是通过为图元的顶点坐标指定恰当的纹理坐标,通过纹理坐标在纹理图中选定特定范围的纹理区域,最后通过纹理坐标和顶点的映射关系,将选定的纹理区域映射到指定的图元上
纹理映射也称为纹理贴图,简单说就是将纹理坐标所指定的纹理区域,映射到顶点坐标对应的渲染区域
纹理坐标是使用纹理坐标系
顶点坐标是使用渲染坐标系或者 OpenGL-ES 坐标系
纹理坐标映射.jpg
4个纹理坐标T0(0,0), T1(0,1), T2(1,1), T3(1,0) 对应的顶点坐标为V0(-1,0.5),V1(-1,-0.5),V2(1,-0.5),V3(1,0.5)
OpenGL-ES 的基本图元是以三角形为单位的,设置绘制两个三角形V0V1V2V0V2V3
当我们调整纹理坐标的顺序保持顶点坐标的顺序不变,T0T1T2T3 ==》T1T2T3T0 绘制后将会得到一个顺时针旋转 90 度为纹理贴图,所以调整纹理坐标和顶点坐标的对应关系可以实现纹理贴图的简单旋转
纹理坐标和纹理像素的映射关系如下:
纹理坐标和纹理像素.png

纹理坐标(0,0)对应纹理像素的第一个元素
纹理坐标(1,0)对应纹理像素的index = width - 1的元素
纹理坐标(0,1)对应纹理像素的index = width* (height - 1) 的元素
纹理坐标(1,1)对应纹理像素的index = width* height - 1 的元素
对应纹理像素区域内的元素都会被采样

纹理映射流程

纹理映射的一般步骤:

  • 生成纹理,编译链接着色器程序
  • 确定纹理坐标和对应的顶点坐标
  • 加载图形数据到纹理,加载顶点坐标和纹理坐标到着色器程序
  • 开始绘制

软件流程如下:
texture_rendering.jpg

对纹理采样的 FragmentShader

"#version 300 es                                     \n"
"precision mediump float;                            \n"
"in vec2 v_texCoord;                                 \n"
"layout(location = 0) out vec4 outColor;             \n"
"uniform sampler2D s_texture;                        \n"
"void main()                                         \n"
"{                                                   \n"
"  outColor = texture( s_texture, v_texCoord );      \n"
"}      

其中 texture 是内置的采样函数,v_texCoord 是顶点着色器传入的纹理坐标,根据纹理坐标进行采样,输出为4向量的 RGBA 值
uniform sampler2D s_texture 是加载后的纹理内容

示例代码:

typedef struct
{// Handle to a program objectGLuint programObject;// Sampler locationGLint samplerLoc;// Texture handleGLuint textureId;
} UserData;// load texture form tga file 
static GLuint CreateSimpleTexture2D()
{// Texture object handleGLuint textureId;// load texture from tga fileconst char* files = "./Huskey.tga";int width;int height;GLubyte* pixels = esLoadTGA(NULL, files, &width, &height);// Use tightly packed dataglPixelStorei(GL_UNPACK_ALIGNMENT, 1);// Generate a texture objectglGenTextures(1, &textureId);// Bind the texture objectglBindTexture(GL_TEXTURE_2D, textureId);// Load the texture notice width height  format must align with real sizeglTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, width, height, 0, GL_RGB, GL_UNSIGNED_BYTE, pixels);// Set the filtering modeglTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);return textureId;
}///
// Initialize the shader and program object
//
static int Init(ESContext *esContext) {UserData *userData = esContext->userData;char vShaderStr[] ="#version 300 es                            \n""layout(location = 0) in vec4 a_position;   \n""layout(location = 1) in vec2 a_texCoord;   \n""out vec2 v_texCoord;                       \n""void main()                                \n""{                                          \n""   gl_Position = a_position;               \n""   v_texCoord = a_texCoord;                \n""}                                          \n";char fShaderStr[] ="#version 300 es                                     \n""precision mediump float;                            \n""in vec2 v_texCoord;                                 \n""layout(location = 0) out vec4 outColor;             \n""uniform sampler2D s_texture;                        \n""void main()                                         \n""{                                                   \n"
//使用内建函数在Fragmanet Shader 中进行纹理采样"  outColor = texture( s_texture, v_texCoord );      \n""}                                                   \n";// Load the shaders and get a linked program objectuserData->programObject = esLoadProgram(vShaderStr, fShaderStr);// Get the sampler locationuserData->samplerLoc = glGetUniformLocation(userData->programObject, "s_texture");// Load the textureuserData->textureId = CreateSimpleTexture2D();glClearColor(1.0f, 1.0f, 1.0f, 0.0f);return TRUE;
}///
// Draw a triangle using the shader pair created in Init()
//
static void Draw(ESContext *esContext)
{UserData *userData = esContext->userData;GLfloat vVertices[] = { -0.5f,  0.5f, 0.0f,  // Position 00.0f,  0.0f,        // TexCoord 0 -0.5f, -0.5f, 0.0f,  // Position 10.0f,  1.0f,        // TexCoord 10.5f, -0.5f, 0.0f,  // Position 21.0f,  1.0f,        // TexCoord 20.5f,  0.5f, 0.0f,  // Position 31.0f,  0.0f         // TexCoord 3};GLushort indices[] = { 0, 1, 2, 0, 2, 3 };// Set the viewportglViewport(0, 0, esContext->width, esContext->height);// Clear the color bufferglClear(GL_COLOR_BUFFER_BIT);// Use the program objectglUseProgram(userData->programObject);// Load the vertex positionglVertexAttribPointer(0, 3, GL_FLOAT,GL_FALSE, 5 * sizeof(GLfloat), vVertices);// Load the texture coordinateglVertexAttribPointer(1, 2, GL_FLOAT,GL_FALSE, 5 * sizeof(GLfloat), &vVertices[3]);glEnableVertexAttribArray(0);glEnableVertexAttribArray(1);// Bind the textureglActiveTexture(GL_TEXTURE0);glBindTexture(GL_TEXTURE_2D, userData->textureId);// Set the sampler texture unit to 0glUniform1i(userData->samplerLoc, 0);glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_SHORT, indices);
}

注意其中的glTexImage2D是用于加载纹理的函数:
void glTexImage2D( GLenum target,GLint level,GLint internalFormat,GLsizei width,GLsizei height,GLint border,
GLenum format,GLenum type,const void * data);

  • GLenum target : 指定texture 类型,一般为 GL_TEXTURE_2D
  • GLint level : 一般设置为0
  • GLint internalFormat : 设置纹理的存储格式 GL_RGB
  • GLsizei width : texture 的宽度
  • GLsizei height : texture 的高度
  • GLint border : 一般设置为0
  • GLenum format : 设置纹理输入图片的存储格式 GL_RGB
  • GLenum type : 设置纹理输入图片的存储格式 GL_RGB
  • const void * data : 指向加载为纹理内容的图片的指针

实际显示效果:
texture_result.jpg

纹理的环绕和过滤方式

纹理坐标的范围通常是从(0, 0) 到 (1, 1),如果设置的坐标超出这个范围,就会有重复的效果,比如 纹理的 S 坐标设置到 2,就表示 S 轴上要重复两次采样纹理,如果 将T 坐标设置为 2,表示 T 轴上重复两次采样纹理,通过设置不同的纹理坐标,就可以产生不同的重复的效果,纹理的环绕方式也可以设置,可以设置的类型如下:
纹理环绕方式.jpg

GL_REPEAT.jpg

GL_MIRRORED_REPEAT.jpg
GL_CLAMP_TO_EDGE.jpg

设置的代码如下:

// GL_REPEAT  GL_CLAMP_TO_EDGE  GL_MIRRORED_REPEAT
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);

纹理的过滤方式

纹理映射的原理是:光栅化之后,图元转换为一个个光栅,每个光栅化之后的 pixel 都会经过 PixelShaderPixelShader 中从纹理采样得到每个 pixel 的颜色,这里就会涉及采样方式的问题,如果图元的分辨率很大,但是关联到图元的纹理很小,就需要进行插值,纹理的过滤方式实际就是纹理到光栅的插值算法
现在只讨论最重要的两种:GL_NEARESTGL_LINEAR

  • GL_NEAREST(也叫邻近过滤,Nearest Neighbor Filtering)是 OpenGL 默认的纹理过滤方式。当设置为 GL_NEAREST 的时候,OpenGL 会选择中心点最接近纹理坐标的那个像素,原理是最近邻插值算法

  • GL_LINEAR(也叫线性过滤,(Bilinear Filtering)它会基于纹理坐标附近的四个纹理像素的值,计算出一个插值,原理是双线性插值算法

邻近过滤算法简单,但是在放大图像的时候,会有严重的马赛克,线性过滤计算过程复杂,但是效果比邻近过滤算法好
GL_NEAREST.jpg

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/79211.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OpenCV计算机视觉实战(2)——环境搭建与OpenCV简介

OpenCV计算机视觉实战(2)——环境搭建与OpenCV简介 0. 前言1. OpenCV 安装与配置1.1 安装 Python-OpenCV1.2 配置开发环境 2. OpenCV 基础2.1 图像读取与显示2.2 图像保存 3. 摄像头实时捕获小结系列链接 0. 前言 OpenCV (Open Source Computer Vision …

ubuntu22.04安装显卡驱动与cuda+cuDNN

背景: 紧接前文:Proxmox VE 8.4 显卡直通完整指南:NVIDIA 2080 Ti 实战。在R740服务器完成了proxmox的安装,并且安装了一张2080ti 魔改22g显存的的显卡。配置完了proxmox显卡直通,并将显卡挂载到了vm 301(…

A2A Python 教程 - 综合指南

目录 • 介绍• 设置环境• 创建项目• 代理技能• 代理卡片• A2A服务器• 与A2A服务器交互• 添加代理功能• 使用本地Ollama模型• 后续步骤 介绍 在本教程中,您将使用Python构建一个简单的echo A2A服务器。这个基础实现将向您展示A2A提供的所有功能。完成本教…

MySQL基础关键_005_DQL(四)

目 录 一、分组函数 1.说明 2.max/min 3.sum/avg/count 二、分组查询 1.说明 2.实例 (1)查询岗位和平均薪资 (2)查询每个部门编号的不同岗位的最低薪资 3.having (1)说明 (2&#xff…

GAMES202-高质量实时渲染(Assignment 2)

目录 作业介绍环境光贴图预计算传输项的预计算Diffuse unshadowedDiffuse shadowedDiffuse Inter-reflection(bonus) 实时球谐光照计算 GitHub主页:https://github.com/sdpyy1 作业实现:https://github.com/sdpyy1/CppLearn/tree/main/games202 作业介绍 物体在不同…

2025年- H21-Lc129-160. 相交链表(链表)---java版

1.题目描述 2.思路 当pa!pb的时候,执行pa不为空,遍历pa链表。执行pb不为空,遍历pb链表。 3.代码实现 // 单链表节点定义 class ListNode {int val;ListNode next;ListNode(int x){valx;nextnull;}}public class H160 {// 主方法…

win10系统安卓开发环境搭建

一 安装jdk 下载jdk17 ,下载路径:https://download.oracle.com/java/17/archive/jdk-17.0.12_windows-x64_bin.exe 下载完毕后,按照提示一步步完成,然后接着创建环境变量, 在cmd控制台输入java -version 验证: 有上面的输出代表jdk安装并配置成功。 二 安装Android stu…

【算法基础】选择排序算法 - JAVA

一、算法基础 1.1 什么是选择排序 选择排序是一种简单直观的排序算法,它的工作原理是:首先在未排序序列中找到最小(或最大)元素,存放到排序序列的起始位置,然后再从剩余未排序元素中继续寻找最小&#xf…

LabVIEW异步调用VI介绍

在 LabVIEW 编程环境里,借助结合异步 VI 调用,并使用 “Open VI Reference” 函数上的 “Enable simultaneous calls on reentrant VIs” 选项(0x40),达成了对多个 VI 调用执行效率的优化。以下将从多方面详细介绍该 V…

Leetcode刷题 | Day50_图论02_岛屿问题01_dfs两种方法+bfs一种方法

一、学习任务 99. 岛屿数量_深搜dfs代码随想录99. 岛屿数量_广搜bfs100. 岛屿的最大面积101. 孤岛的总面积 第一类DFS(主函数中处理第一个节点,DFS处理相连节点): 主函数中先将起始节点标记为已访问DFS函数中不处理起始节点&…

深入理解网络安全中的加密技术

1 引言 在当今数字化的世界中,网络安全已经成为个人隐私保护、企业数据安全乃至国家安全的重要组成部分。随着网络攻击的复杂性和频率不断增加,保护敏感信息不被未授权访问变得尤为关键。加密技术作为保障信息安全的核心手段,通过将信息转换为…

旧版本NotionNext图片失效最小改动解决思路

旧版本NotionNext图片失效最小改动解决思路 契机 好久没写博客了,最近在notion写博客的时候发现用notionNext同步到个人网站时,图片无法预览。猜测是notion加了防盗链措施,去notionNext官方github上寻找解决方案,需要升级到4.8.…

深度学习笔记40_中文文本分类-Pytorch实现

🍨 本文为🔗365天深度学习训练营 中的学习记录博客🍖 原作者:K同学啊 | 接辅导、项目定制 一、我的环境 1.语言环境:Python 3.8 2.编译器:Pycharm 3.深度学习环境: torch1.12.1cu113torchvision…

010302-oss_反向代理_负载均衡-web扩展2-基础入门-网络安全

文章目录 1 OSS1.1 什么是 OSS 存储?1.2 OSS 核心功能1.3 OSS 的优势1.4 典型使用场景1.5 如何接入 OSS?1.6 注意事项1.7 cloudreve实战演示1.7.1 配置cloudreve连接阿里云oss1.7.2 常见错误1.7.3 安全测试影响 2 反向代理2.1 正向代理和反向代理2.2 演示…

【 Node.js】 Node.js安装

下载 下载 | Node.js 中文网https://nodejs.cn/download/ 安装 双击安装包 点击Next 勾选使用许可协议,点击Next 选择安装位置 点击Next 点击Next 点击Install 点击Finish 完成安装 添加环境变量 编辑【系统变量】下的变量【Path】添加Node.js的安装路径--如果…

Python基本语法(自定义函数)

自定义函数 Python语言没有子程序,只有自定义函数,目的是方便我们重复使用相同的一 段程序。将常用的代码块定义为一个函数,以后想实现相同的操作时,只要调用函数名就可以了,而不需要重复输入所有的语句。 函数的定义…

OpenGL-ES 学习(11) ---- EGL

目录 EGL 介绍EGL 类型和初始化EGL初始化方法获取 eglDisplay初始化 EGL选择 Config构造 Surface构造 Context开始绘制 EGL Demo EGL 介绍 OpenGL-ES 是一个操作GPU的图像API标准,它通过驱动向 GPU 发送相关图形指令,控制图形渲染管线状态机的运行状态&…

极简5G专网解决方案

极简5G专网解决方案 利用便携式即插即用私有 5G 网络提升您的智能创新。为您的企业提供无缝、安全且可扩展的 5G 解决方案。 提供极简5G专网解决方案 Mantiswave Network Private Limited 提供全面的 5G 专用网络解决方案,以满足您企业的独特需求。我们创新的“…

html:table表格

表格代码示例&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title> </head> <body><!-- 标准表格。 --><table border"5"cellspacing&qu…