【OpenGL实现 03】纹理贴图原理和实现

目录

  • 一、说明
  • 二、纹理贴图原理
    • 2.1 纹理融合原理
    • 2.2 UV坐标原理
  • 三、生成纹理对象
    • 3.1 需要在VAO上绑定纹理坐标
    • 3.2 纹理传递
    • 3.3 纹理buffer生成
  • 四、代码实现:
  • 五、着色器
    • 4.1 片段
    • 4.2 顶点
  • 五、后记

一、说明

本篇叙述在画出图元的时候,如何贴图纹理图片。和纹理坐标的原理实现,以及纹理如何生成,和如何传递进入着色器。对以上进行解说,并附上源代码。

二、纹理贴图原理

2.1 纹理融合原理

1 融合blend的条件

  • 要使用OpenGL的混合功能,只需要调用:glEnable(GL_BLEND);即可。
  • 要关闭OpenGL的混合功能,只需要调用:glDisable(GL_BLEND);即可。
  • 注意:只有在RGBA模式下,才可以使用混合功能,颜色索引模式下是无法使用混合功能的。

2 glBlendFunc() 融合函数

OpenGL会把源颜色和目标颜色各自取出,并乘以一个系数(源颜色乘以的系数称为“源因子”,目标颜色乘以的系数称为“目标因子”),然后相加,这样就得到了新的颜色。
如:
C o l o r n e w = α C o l o r f r e g + ( 1 − α ) C o l o r t e x t i r e        ( α ∈ [ 0 , 1 ] ) Color_{new} =\alpha Color_{freg}+ (1-\alpha) Color_{textire} \;\;\; ( \alpha \in[0,1]) Colornew=αColorfreg+1αColortextire(α[0,1])

除此之外,还有新的算法。新版本的OpenGL可以设置运算方式,包括加、减、取两者中较大的、取两者中较小的、逻辑运算等)下表:

选择宏代码意义
GL_ZERO表示使用0.0作为因子,实际上相当于不使用这种颜色参与混合运算。
GL_ONE表示使用1.0作为因子,实际上相当于完全的使用了这种颜色参与混合运算。
GL_SRC_ALPHA表示使用源颜色的alpha值来作为因子。
GL_DST_ALPHA表示使用目标颜色的alpha值来作为因子。
GL_ONE_MINUS_SRC_ALPHA表示用1.0减去源颜色的alpha值来作为因子(1-alpha)。
GL_ONE_MINUS_DST_ALPHA表示用1.0减去目标颜色的alpha值来作为因子。
GL_SRC_COLOR(把源颜色的四个分量分别作为因子的四个分量)
GL_ONE_MINUS_SRC_COLORxxx
GL_DST_COLORxxx
GL_ONE_MINUS_DST_COLOR前两个在OpenGL旧版本中只能用于设置目标因子,后两个在OpenGL 旧版本中只能用于设置源因子。新版本的OpenGL则没有这个限制,并且支持新的
GL_CONST_COLOR(设定一种常数颜色,将其四个分量分别作为 因子的四个分量)GL_ONE_MINUS_CONST_COLOR
GL_CONST_ALPHAxx
GL_ONE_MINUS_CONST_ALPHAxxx
GL_SRC_ALPHA_SATURATE新版本的OpenGL还允许颜色的alpha 值和RGB值采用不同的混合因子

举例来说: 如果设置了glBlendFunc(GL_ONE, GL_ZERO);,则表示完全使用源颜色,完全不使用目标颜色,因此画面效果和不使用混合的时候一致(当然效率可能会低一点点)。

如果设置了glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);则表示源颜色乘以自身的alpha 值,目标颜色乘以1.0减去源颜色的alpha值,这样一来,源颜色的alpha值越大,则产生的新颜色中源颜色所占比例就越大,而目标颜色所占比例则减 小。这种情况下,我们可以简单的将源颜色的alpha值理解为“不透明度”。这也是混合时最常用的方式。

如果设置了glBlendFunc(GL_ONE, GL_ONE);,则表示完全使用源颜色和目标颜色,最终的颜色实际上就是两种颜色的简单相加。例如红色(1,0,0)和绿色(0,1,0)相加得到(1,1,0),结果为黄色。

注意: 所 谓源颜色和目标颜色,是跟绘制的顺序有关的。假如先绘制了一个红色的物体,再在其上绘制绿色的物体。则绿色是源颜色,红色是目标颜色。如果顺序反过来,则 红色就是源颜色,绿色才是目标颜色。在绘制时,应该注意顺序,使得绘制的源颜色与设置的源因子对应,目标颜色与设置的目标因子对应。不要被混乱的顺序搞晕 。

2.2 UV坐标原理

这个坐标不是孤立的,而是与被画图元的摆放有关。比如,下图:

在这里插入图片描述
当出现如下配置时:
vertices = (
-0.5, -0.5, 0.0, 1.0, 0.0, 0.0, 0.0, 1.0,
0.5, -0.5, 0.0, 0.0, 1.0, 0.0, 1.0, 1.0,
0.0, 1.0, 0.0, 0.0, 0.0, 1.0, 1.0, 0.0,)

在这里插入图片描述
这是一个三角形对另一个三角形的映射。即:
S 1 → S 2 S_1 \rightarrow S_2 S1S2
这种映射很容易实现,比如:
( a b c d e f 0 0 1 ) ( x 0 y 0 1 ) = ( U 0 V 0 1 ) \begin{pmatrix} a & b& c\\ d & e& f\\ 0 & 0 & 1 \end{pmatrix} \begin{pmatrix} x_0\\ y_0\\ 1 \end{pmatrix} = \begin{pmatrix} U_0\\ V_0\\ 1 \end{pmatrix} ad0be0cf1 x0y01 = U0V01
( a b c d e f 0 0 1 ) ( x 1 y 1 1 ) = ( U 1 V 1 1 ) \begin{pmatrix} a & b& c\\ d & e& f\\ 0 & 0 & 1 \end{pmatrix} \begin{pmatrix} x_1\\ y_1\\ 1 \end{pmatrix} = \begin{pmatrix} U_1\\ V_1\\ 1 \end{pmatrix} ad0be0cf1 x1y11 = U1V11
( a b c d e f 0 0 1 ) ( x 2 y 2 1 ) = ( U 2 V 2 1 ) \begin{pmatrix} a & b& c\\ d & e& f\\ 0 & 0 & 1 \end{pmatrix} \begin{pmatrix} x_2\\ y_2\\ 1 \end{pmatrix} = \begin{pmatrix} U_2\\ V_2\\ 1 \end{pmatrix} ad0be0cf1 x2y21 = U2V21

以上三个方程联立后,就能确定 a , b , c , d , e , f a,b,c,d,e,f a,b,c,d,e,f 从而对任意点插值:
顺便提一下,以上名称叫“仿射变换”,听不懂的,就跟我慢慢学吧!

三、生成纹理对象

3.1 需要在VAO上绑定纹理坐标

glEnableVertexAttribArray(2)
glVertexAttribPointer(2, 2, GL_FLOAT, GL_FALSE, 32, ctypes.c_void_p(24))

3.2 纹理传递

glUniform1i(glGetUniformLocation(self.shader, "imageTexture"), 0)

注意:将Uniform类型传入shade需要执行两步:
1)查找到GPU中名称为"imageTexture"的变量地址
2)将cpu纹理数组传递进GPU,glUniform1i

3.3 纹理buffer生成

    self.texture = glGenTextures(1)
    glBindTexture(GL_TEXTURE_2D, self.texture)
    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_MIN_FILTER, GL_NEAREST)
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR)
    image = pg.image.load(filepath).convert_alpha()
    image_width,image_height = image.get_rect().size
    img_data = pg.image.tostring(image,'RGBA')
    glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA,image_width,image_height,0,GL_RGBA,GL_UNSIGNED_BYTE,img_data)
    glGenerateMipmap(GL_TEXTURE_2D)

四、代码实现:

import pygame as pg
from OpenGL.GL import *
from OpenGL.GL.shaders import compileProgram,compileShader
import numpy as np

def create_shader(vertex_filepath: str, fragment_filepath: str) -> int:
    """
        Compile and link shader modules to make a shader program.
        Parameters:
            vertex_filepath: path to the text file storing the vertex
                            source code
            fragment_filepath: path to the text file storing the
                                fragment source code
        Returns:
            A handle to the created shader program
    """

    with open(vertex_filepath,'r') as f:
        vertex_src = f.readlines()

    with open(fragment_filepath,'r') as f:
        fragment_src = f.readlines()
    
    shader = compileProgram(compileShader(vertex_src, GL_VERTEX_SHADER),
                            compileShader(fragment_src, GL_FRAGMENT_SHADER))
    return shader

class App:
    """
        For now, the app will be handling everything.
        Later on we'll break it into subcomponents.
    """
    def __init__(self):
        """ Initialise the program """
        self._set_up_pygame()
        self._set_up_timer()
        self._set_up_opengl()
        self._create_assets()
        self._set_onetime_uniforms()
    
    def _set_up_pygame(self) -> None:
        """
            Initialize and configure pygame.
        """
        pg.init()
        pg.display.gl_set_attribute(pg.GL_CONTEXT_MAJOR_VERSION, 3)
        pg.display.gl_set_attribute(pg.GL_CONTEXT_MINOR_VERSION, 3)
        pg.display.gl_set_attribute(pg.GL_CONTEXT_PROFILE_MASK, pg.GL_CONTEXT_PROFILE_CORE)
        pg.display.set_mode((640,480), pg.OPENGL|pg.DOUBLEBUF)

    def _set_up_timer(self) -> None:
        """
            Set up the app's timer.
        """
        self.clock = pg.time.Clock()
    
    def _set_up_opengl(self) -> None:
        """
            Configure any desired OpenGL options
        """
        glClearColor(0.1, 0.2, 0.2, 1)
        glEnable(GL_BLEND)
        glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA)
    
    def _create_assets(self) -> None:
        """
            Create all of the assets needed for drawing.
        """

        self.triangle = Triangle()
        self.wood_texture = Material("gfx/land.png")
        self.shader = create_shader(
            vertex_filepath = "shaders/vertex.txt", fragment_filepath = "shaders/fragment.txt")
    
    def _set_onetime_uniforms(self) -> None:
        """
            Some shader data only needs to be set once.
        """
        glUseProgram(self.shader)
        glUniform1i(glGetUniformLocation(self.shader, "imageTexture"), 0)

    def run(self) -> None:
        """ Run the app """
        running = True
        while (running):
            #check events
            for event in pg.event.get():
                if (event.type == pg.QUIT):
                    running = False
            #refresh screen
            glClear(GL_COLOR_BUFFER_BIT)

            glUseProgram(self.shader)
            self.wood_texture.use()
            self.triangle.arm_for_drawing()
            self.triangle.draw()

            pg.display.flip()

            #timing
            self.clock.tick(60)

    def quit(self) -> None:
        """ cleanup the app, run exit code """

        self.triangle.destroy()
        self.wood_texture.destroy()
        glDeleteProgram(self.shader)
        pg.quit()

class Triangle:
    """
        Yep, it's a triangle.
    """
    def __init__(self):
        """
            Initialize a triangle.
        """
        # x, y, z, r, g, b, s, t
        vertices = (
            -0.5, -0.5, 0.0, 1.0, 0.0, 0.0, 0.0, 1.0,
             0.5, -0.5, 0.0, 0.0, 1.0, 0.0, 1.0, 1.0,
             0, 1, 0.0, 0.0, 0.0, 1.0, 1, 0.0,
            -0.5, 0.0, 0.0, 1.0, 0.0, 1.0,  0.0, 1.0,
             0.25, 0.5, 0.0, 1.0, 0.0, 1.0, 1.0, 1.0,
            -0.25, 0.5, 0.0, 1.0, 0.0, 1.0, 1.0, 0.0 )

        vertices = np.array(vertices, dtype=np.float32)

        self.vertex_count = 3
        self.vao = glGenVertexArrays(1)
        glBindVertexArray(self.vao)
        self.vbo = glGenBuffers(1)
        glBindBuffer(GL_ARRAY_BUFFER, self.vbo)
        glBufferData(GL_ARRAY_BUFFER, vertices.nbytes, vertices, GL_STATIC_DRAW)

        glEnableVertexAttribArray(0)
        glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 32, ctypes.c_void_p(0))
        
        glEnableVertexAttribArray(1)
        glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, 32, ctypes.c_void_p(12))

        glEnableVertexAttribArray(2)
        glVertexAttribPointer(2, 2, GL_FLOAT, GL_FALSE, 32, ctypes.c_void_p(24))

    def arm_for_drawing(self) -> None:
        """
            Arm the triangle for drawing.
        """
        glBindVertexArray(self.vao)
    
    def draw(self) -> None:
        """
            Draw the triangle.
        """

        glDrawArrays(GL_TRIANGLES, 0, self.vertex_count)

    def destroy(self) -> None:
        """
            Free any allocated memory.
        """
        
        glDeleteVertexArrays(1,(self.vao,))
        glDeleteBuffers(1,(self.vbo,))

class Material:
    """
        A basic texture.
    """
    def __init__(self, filepath: str):
        """
            Initialize and load the texture.

            Parameters:

                filepath: path to the image file.
        """
        self.texture = glGenTextures(1)
        glBindTexture(GL_TEXTURE_2D, self.texture)
        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_MIN_FILTER, GL_NEAREST)
        glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR)
        image = pg.image.load(filepath).convert_alpha()
        image_width,image_height = image.get_rect().size
        img_data = pg.image.tostring(image,'RGBA')
        glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA,image_width,image_height,0,GL_RGBA,GL_UNSIGNED_BYTE,img_data)
        glGenerateMipmap(GL_TEXTURE_2D)

    def use(self) -> None:
        """
            Arm the texture for drawing.
        """
        glActiveTexture(GL_TEXTURE0)
        glBindTexture(GL_TEXTURE_2D,self.texture)

    def destroy(self) -> None:
        """
            Free the texture.
        """

        glDeleteTextures(1, (self.texture,))

my_app = App()
my_app.run()
my_app.quit()

五、着色器

4.1 片段

着色器需要将纹理图片的具体数据带入进来,是通过uniform sampler2D类型数据带入的。最后的颜色,是顶点固有color和imageTexture融合的结果。
fragment.txt

#version 330 core

in vec3 fragmentColor;
in vec2 fragmentTexCoord;

out vec4 color;

uniform sampler2D imageTexture;

void main()
{
    color = 0.4*vec4(fragmentColor, 1.0) + 0.6*texture(imageTexture, fragmentTexCoord);
}

4.2 顶点

顶点输入时候,顺便将该顶点所对应的纹理图片坐标带入。

#version 330 core

layout (location=0) in vec3 vertexPos;
layout (location=1) in vec3 vertexColor;
layout (location=2) in vec2 vertexTexCoord;

out vec3 fragmentColor;
out vec2 fragmentTexCoord;

void main()
{
    gl_Position = vec4(vertexPos, 1.0);
    fragmentColor = vertexColor;
    fragmentTexCoord = vertexTexCoord;
}

五、后记

我承认有一些细节没有写出,如果读者需要更详细资料,请从资源中下载源码:
xxxxx

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

NLP_文本张量表示方法_2(代码示例)

目标 了解什么是文本张量表示及其作用.文本张量表示的几种方法及其实现. 1 文本张量表示 将一段文本使用张量进行表示,其中一般将词汇为表示成向量,称作词向量,再由各个词向量按顺序组成矩阵形成文本表示. ["人生", "该&q…

【深度学习笔记】稠密连接网络(DenseNet)

注:本文为《动手学深度学习》开源内容,部分标注了个人理解,仅为个人学习记录,无抄袭搬运意图 5.12 稠密连接网络(DenseNet) ResNet中的跨层连接设计引申出了数个后续工作。本节我们介绍其中的一个&#xf…

LInux-多线程基础概念

文章目录 前言预备页表详解缺页中断页表的映射 一、多线程是什么?轻量级进程 二、Pthread库pthread_create 前言 从本章的多线程开始,我们开始进入Linux系统的尾声,所以,在学习多线程的过程中,我们也会逐步对之前的内…

第十五课 PCB保姆级规则设置(四)

1.常用的规则设置 2.设置完规则后,重新设置规则检查 将不设置的规则,不勾选 3.布线规则 (1)先走短线(模块之内的线) 模块之内的线需要连接,长线的话提前打孔(占位置)…

微信小程序开发系列(十五)·bind 绑定的事件(冒泡事件)、catch 绑定的事件(非冒泡事件)

目录 1. 事件分类 1.1 bind 绑定的事件(冒泡事件) 1.2 catch 绑定的事件(非冒泡事件) 1.3 代码 1. 事件分类 事件分为 冒泡事件 和 非冒泡事件: 冒泡事件:当一个组件的事件被触发后,…

浅谈字典攻击

一、前言 字典攻击是一种常见的密码破解方法,它使用预先编制的字典文件作为攻击字典,通过尝试猜测密码的方式来破解密码。下面是一个关于字典攻击的博客,希望能够为您了解字典攻击提供帮助。 二、字典攻击概述 字典攻击是一种密码破解方法&…

微信小程序 弹窗禁止底部页面滚动

解决方法: 在弹窗上给添加catchtouchmove catchtouchmovereturn

个人商城系统开源(展示宣传)

原文地址:个人商城系统开源(展示宣传) - Pleasure的博客 下面是正文内容: 前言 由于近期实在没有什么话题可写和一些有趣的项目教程可以分享。所以我只能决定将我自己亲手编写的一个迷你商城系统进行开源。 也就是放在我博客右边…

VR虚拟生态探索介绍|元宇宙文旅|VR设备价格

虚拟现实(VR)虚拟生态探索是一种通过VR技术让用户体验并探索虚拟环境的活动。在这样的环境中,用户可以通过头戴式显示器和其他感知设备沉浸在一个虚拟的世界中,仿佛身临其境。 在VR虚拟生态探索中,用户可以探索各种不同…

数据库备份.....

一.环境准备 数据库备份,数据库为school,素材如下 >create database school; >use school1.创建student和score表CREATE TABLE student ( id INT(10) NOT NULL UNIQUE PRIMARY KEY , name VARCHAR(20) NOT NULL , sex VARCHAR(4) , bi…

【PCL】(二十五)基于Min-Cut的点云分割

Min-Cut Based Segmentation 提出这个方法的论文&#xff1a;Min-Cut Based Segmentation of Point Clouds 最大流最小割问题 min_cut_segmentation.cpp #include <iostream> #include <vector> #include <pcl/io/pcd_io.h> #include <pcl/point_types…

基于ZEM系列台式扫描电镜和EDS能谱仪的岩石元素成分分析研究

上一期里我们对几种不同种类的岩石在特定作用下的破坏特征进行了表征&#xff0c;感兴趣的同学可以阅读《从宏观到微观——泽攸科技ZEM系列台式扫描电镜在岩石分析中的应用》。岩石是地球表层和地壳的基本组成物质&#xff0c;其成因和成岩过程复杂多样&#xff0c;岩石的元素组…

PythonStudio 控件使用常用方式(十)TLabel和TLinkLabel

PythonStudio是一个极强的开发Python的IDE工具&#xff0c;它使用的是Delphi的控件&#xff0c;常用的内容是与Delphi一致的。但是相关文档并一定完整。现在我试试能否逐步把它的控件常用用法写一点点&#xff0c;也作为PythonStudio的参考。 Tlabel是最常用的控件&#xff0c…

【论文阅读】Segment Anything论文梳理

Abstract 我们介绍了Segment Anything&#xff08;SA&#xff09;项目&#xff1a;新的图像分割任务、模型和数据集。高效的数据循环采集&#xff0c;使我们建立了迄今为止最大的分割数据集&#xff0c;在1100万张图像中&#xff0c;共超过10亿个掩码。 该模型被设计和训练为可…

【最详细的python教程】Python基本数据类型及运算符

&#x1f383;Python基本数据类型 &#x1f38b;整数** 整数是表述整数的数值&#xff0c;没有小数部分。在Python中&#xff0c;整数包括正整数、负整数和0&#xff0c;在Python中数值超过计算机自身的计算功能时会自动转用高精度计算。整数类型包括二进制、八进制、十进制和…

摘花生c++

题目 Hello Kitty想摘点花生送给她喜欢的米老鼠。 她来到一片有网格状道路的矩形花生地(如下图)&#xff0c;从西北角进去&#xff0c;东南角出来。 地里每个道路的交叉点上都有种着一株花生苗&#xff0c;上面有若干颗花生&#xff0c;经过一株花生苗就能摘走该它上面所有的…

指针易错点(超详细)

&#x1f4cc; 博客主页 爆打维c 本文将介绍指针基本知识点及易错点&#xff0c;刚入门学习c语言的小伙伴们可以收藏起来&#xff0c;方便找到。 目录 一、指针是什么&#xff1f; 1.const修饰指针 总结: 2.野指针 野指针成因: 3.指针数组与数组指针的区别 3.1指针数…

数据分析师必备:五款数据可视化工具对比与推荐

在数字化时代&#xff0c;数据可视化产品成为了企业和个人进行数据分析、信息呈现的重要工具。市面上涌现了众多数据可视化产品&#xff0c;它们各具特色&#xff0c;功能各异。本文为大家简要介绍五款市面上热门的数据可视化产品。 一、Tableau Tableau是一款功能强大的数据…

精读《React Conf 2019 - Day2》

1 引言 这是继 精读《React Conf 2019 - Day1》 之后的第二篇&#xff0c;补充了 React Conf 2019 第二天的内容。 2 概述 & 精读 第二天的内容更为精彩&#xff0c;笔者会重点介绍比较干货的部分。 Fast refresh Fast refresh 是更好的 react-hot-loader 替代方案&am…

mysql日常优化的总结

文章目录 一、数据表结构相关优化建字段类型注意事项1. int类型的选择2.varchar、char、text类型3.date、datetime、timestamp类型 表规划1. 垂直分表2. 水平分表 二、查询语句优化1.对于字段多的表&#xff0c;避免使用SELECT *2.避免使用!操作符3.避免使用null做条件4.like查…