在第 5 篇文章中,我们成功加载了 fbx 模型,并且做了 MVP 变换,将立方体按照透视投影渲染了出来。但是当时只是随机给顶点颜色,并且默认 fbx 文件里只有一个 mesh,这次我们来加载一个柴犬模型,并且给模型贴图,模型可以从 sketchfab 下载

本文没有涉及到理论解释,更多的是代码实践。

完整代码在 https://github.com/MangoWAY/CGLearner/tree/v0.3 tag v0.3

1. 创建纹理,加载图片

我们来封装一个 Texture 类用来加载图片,创建、bind 纹理,加载图片我用的是 pillow 库。

from OpenGL import GL as gl
from PIL import Image
import numpy as np
class Texture:
COUNT = 0
def __init__(self) -> None:
self.texid = -1
self.count = -1 def create(self):
self.texid = gl.glGenTextures(1) def load_from_path(self, path: str):
gl.glActiveTexture(gl.GL_TEXTURE0 + Texture.COUNT)
self.count = Texture.COUNT
Texture.COUNT +=1
gl.glBindTexture(gl.GL_TEXTURE_2D, self.texid)
# Set the texture wrapping parameters
gl.glTexParameteri(gl.GL_TEXTURE_2D, gl.GL_TEXTURE_WRAP_S, gl.GL_REPEAT)
gl.glTexParameteri(gl.GL_TEXTURE_2D, gl.GL_TEXTURE_WRAP_T, gl.GL_REPEAT)
# Set texture filtering parameters
gl.glTexParameteri(gl.GL_TEXTURE_2D, gl.GL_TEXTURE_MIN_FILTER, gl.GL_LINEAR)
gl.glTexParameteri(gl.GL_TEXTURE_2D, gl.GL_TEXTURE_MAG_FILTER, gl.GL_LINEAR)
# load image
image = Image.open(path)
img_data = np.array(list(image.getdata()), np.uint8)
gl.glTexImage2D(gl.GL_TEXTURE_2D,
0,
gl.GL_RGB,
image.width,
image.height,
0,
gl.GL_RGB,
gl.GL_UNSIGNED_BYTE,
img_data)
gl.glGenerateMipmap(gl.GL_TEXTURE_2D) def bind(self):
gl.glActiveTexture(gl.GL_TEXTURE0 + self.count)
gl.glBindTexture(gl.GL_TEXTURE_2D, self.texid)

2. UV 采样

在之前的文章中,我们基本只用到了顶点的位置信息,这次我们需要用到顶点的 uv 坐标,我们根据 uv 坐标对纹理进行采样,获取当前的颜色。如下,在之前封装的模型加载类里,用 pyassimp 获取 uv 坐标。

# model_importer.py
...
def load_mesh(self, path: str):
scene = pyassimp.load(path)
mmeshes = []
for mesh in scene.meshes:
...
# 获取 uv 坐标
mmesh.uvs = mesh.texturecoords.squeeze(0)
...
return mmeshes
...

有了 uv 以后,我们需要将它放到我们的顶点数组里,然后正确设置长度、偏移等等,和位置、法线等数据类似。有一点需要注意一下,图片的坐标系原点一般在左上,而 uv 坐标的原点在左下,因此需要 y 方向需要翻转一下。vert 如下,我们新加一个 uv 的顶点属性,然后将它传递到 frag shader 中。在 frag 中翻转一下 y,然后采样纹理。

// vert
#version 330 core
...
layout(location = 3) in vec2 aUV;
out vec3 c;
out vec2 uv;
uniform mat4 u_mvp;
void main(){
gl_Position = u_mvp * vec4(aPos,1.0);
c = aColor;
uv = aUV;
}
// frag
#version 330 core
out vec4 color;
in vec3 c;
in vec2 uv;
uniform sampler2D ourTexture;
void main(){
...
vec2 uv1 = vec2(uv.x,1.0-uv.y);
color = texture(ourTexture, uv1);
}

3. 绘制多个网格

这个柴犬模型里有 3 个网格,我们需要绘制 3 个网格,因此我们需要修改一下之前主函数的逻辑,之前是默认加载的第一个网格,现在需要加载每一个网格,然后创建 VAO、VBO、EBO 等渲染数据,然后加载纹理资源,最后在渲染循环中依次渲染。

# main.py
...
verts = []
indes = []
renderData = []
for mesh in meshes:
vert = []
for i in range(len(mesh.vertices)):
if i % 3 == 0:
vert.extend([mesh.vertices[i],mesh.vertices[i + 1],mesh.vertices[i + 2]])
vert.extend([mesh.normals[i],mesh.normals[i + 1],mesh.normals[i + 2]])
vert.extend([random.random(),random.random(),random.random()])
vert.extend([mesh.uvs[int(i/3),0],mesh.uvs[int(i/3),1]])
verts.append(vert)
inde = mesh.subMeshes[0].indices
indes.append(inde)
data = RendererData()
data.build_data([desp,desp1,desp2,desp3],vert, inde)
renderData.append(data)
...
tex = Texture()
tex.create()
tex.load_from_path("default_Base_Color.png")
tex.bind() while (...):
...
for data in renderData:
data.use()
data.draw()
data.unuse()
...

我们可以调一调之前定义的 Transform 的位置、角度,或者相机的角度等,渲染的结果如下:

4. 总结

  • 加载 uv 坐标传递到 shader 中;
  • 利用 pyopengl 加载纹理贴图;
  • 渲染多个网格数据;

[CG从零开始] 6. 加载一个柴犬模型学习UV贴图的更多相关文章

  1. jQuery加载一个html页面到指定的div里

    一.jQuery加载一个html页面到指定的div里 把a.html里面的某一部份的内容加载到b.html的一个div里.比如:加载a.html里面的<div id=“row"> ...

  2. 无法加载一个或多个请求的类型。有关更多信息,请检索 LoaderExceptions 属性。

    新建一个MVC4的项目,引用DAL后,将DAL的连接字符串考入: <connectionStrings>     <add name="brnmallEntities&qu ...

  3. “无法加载一个或多个请求的类型。有关更多信息,请检索 LoaderExceptions 属性 “之解决

    今天在学习插件系统设计的时候遇到一个问题:“System.Reflection.ReflectionTypeLoadException: 无法加载一个或多个请求的类型. 于是百度一下,很多内容都差不多 ...

  4. Tomcat启动时自动加载一个类

    有时候在开发Web应用的时候,需要tomcat启动后自动加载一个用户的类,执行一些初始化方法,如从数据库中加载业务字典到内存中,因此需要在tomcat启动时就自动加载一个类,或运行一个类的方法. 可以 ...

  5. Android 编程下 WebView 加载一个网页如何得到网页的 Cookie 值

    http://www.cnblogs.com/sunzn/archive/2013/04/03/2998113.html mWebView.setWebViewClient(new MyWebView ...

  6. JavaWeb 服务启动时,在后台启动加载一个线程

    JavaWeb 服务启动时,在后台启动加载一个线程. 目前,我所掌握的一共有两种方法,第一种是监听(Listener),第二种是配置随项目启动而启动的Servlet. 下面对这两种方法做一简单的介绍, ...

  7. 如何在tomcat启动时自动加载一个类

    有时候在开发web应用的时候,需要tomcat启动后自动加载一个用户的类,执行一些初始化方法,如从数据库中加载业务字典到内存中,因此需要在tomcat启动时就自动加载一个类,或运行一个类的方法. 可以 ...

  8. tomcat启动时自动加载一个类 MyServletContextListener

    目的: 我们知道在tomcat启动后,需要页面请求进行驱动来执行操作接而响应.我们希望在tomcat启动的时候能够自动运行一个后台线程,以处理我们需要的一些操作.因此需要tomcat启动时就自动加载一 ...

  9. wpf prism4 出现问题:无法加载一个或多个请求的类型。有关更多信息,请检索 LoaderExceptions 属性。

    WPF Prism 框架 程序 出现 问题: 无法加载一个或多个请求的类型.有关更多信息,请检索 LoaderExceptions 属性. 1.开始以为是配置的问题,找了半天,最后原来是有个依赖类库没 ...

随机推荐

  1. JavaWeb--HTTP与Maven

    前言 Java Web 其实就是一个技术的总和,把Web看成一个容器而已主要使用JavaEE技术来实现.在加上各种中间件. 整个javaWeb阶段的内容通过实际的案例贯穿学习, 所涉及到的技术知识点会 ...

  2. SkiaSharp 之 WPF 自绘 投篮小游戏(案例版)

    此案例主要是针对光线投影法碰撞检测功能的示例,顺便做成了一个小游戏,很简单,但是,效果却很不错. 投篮小游戏 规则,点击投篮目标点,就会有一个球沿着相关抛物线,然后,判断是否进入篮子里,其实就是一个矩 ...

  3. 大数据平台迁移实践 | Apache DolphinScheduler 在当贝大数据环境中的应用

    大家下午好,我是来自当贝网络科技大数据平台的基础开发工程师 王昱翔,感谢社区的邀请来参与这次分享,关于 Apache DolphinScheduler 在当贝网络科技大数据环境中的应用. 本次演讲主要 ...

  4. Taurus.MVC 微服务框架 入门开发教程:项目部署:3、微服务应用程序版本升级:全站升级和局部模块升级。

    系列目录: 本系列分为项目集成.项目部署.架构演进三个方向,后续会根据情况调整文章目录. 本系列第一篇:Taurus.MVC V3.0.3 微服务开源框架发布:让.NET 架构在大并发的演进过程更简单 ...

  5. 论文解读(GATv2)《How Attentive are Graph Attention Networks?》

    论文信息 论文标题:How Attentive are Graph Attention Networks?论文作者:Shaked Brody, Uri Alon, Eran Yahav论文来源:202 ...

  6. ENSP 与VMware workstation 同时安装会出现AR40 或者VMware蓝屏

    前言: 你已经安装了VMware workstation,ENSP,Virtualbox,并且关闭了hyper-v,还会出现报错或者VMware蓝屏. #此处补充一条命令systeminfo 如果你看 ...

  7. 我开源了一个Go学习仓库|笔记预览

    前言 大半个月前我参与了字节后端面试,未通过第四面,面试总结写在了这篇文章: https://juejin.cn/post/7132712873351970823 在此文的末尾,我写到为了全面回顾Go ...

  8. ASP.NET Core 6框架揭秘实例演示[33]:异常处理高阶用法

    NuGet包"Microsoft.AspNetCore.Diagnostics"中提供了几个与异常处理相关的中间件,我们可以利用它们将原生的或者定制的错误信息作为响应内容发送给客户 ...

  9. 【Java】学习路径51-线程组

    平时创建线程的时候,系统会默认为线程分组. 我们可以使用 ThreadGroup tg1 = t1.getThreadGroup(); 取得t1的线程组对象. 然后使用getName获得线程组名称. ...

  10. SpringBoot中maven项目Plugins里resources报红

    错误原因:刚开始下载依赖时下载错误导致的 参考:解决idea中maven plugins标红的问题 - 走看看 (zoukankan.com) 如果还不行: 就根据上面提示的地址找到maven的配置包 ...