WebGL树形结构的模型渲染流程
今天和大家分享的是webgl渲染树形结构的流程。用过threejs,babylonjs的同学都知道,一个大模型都是由n个子模型拼装而成的,那么如何依次渲染子模型,以及渲染每个子模型在原生webgl中的流程是怎样的呢,我就以osg框架为原本,为同学们展示出来。
首先介绍osg框架,该框架是基于openGL的几何引擎框架,目前我的工作是将其翻译成为webgl的几何引擎,在这个过程中学习webgl原生架构的原理和工程构造方式,踩了各种坑,每次爬出坑都觉得自己又强大了一点,呵。
闲话少叙,切入正题,首先我们要明确一个渲染流程,那就是webgl到底是怎么将模型绘制到canvas画布中去的,这就牵扯到我之前的一片文章《原生WebGL场景中绘制多个圆锥圆柱》,链接地址https://www.cnblogs.com/ccentry/p/9864847.html,这篇文章讲述的是用原生webgl向canvas中持续绘制多个模型,但这篇文章的局限性在于,他重复使用了同一组shader(顶点shader,片段shader),并且多个模型也不存在父子关系,这就导致了局部坐标系和全局坐标系的紊乱。今天我们就来弥补这篇文章的不足之处。
按部就班,我们先讨论的是webgl渲染单个模型的过程,首先我们构造着色器,请看下面着色器代码
attribute vec3 position;
attribute vec3 normal;
attribute vec4 color;
uniform mat4 mvpMatrix;
uniform mat4 invMatrix;
uniform vec3 lightDirection;
uniform vec4 ambientColor;
varying vec4 vColor;
uniform float lightS; void main(void){
vec3 invLight = normalize(invMatrix * vec4(lightDirection, )).xyz;
float diffuse = clamp(dot(normal, invLight), 0.0, 1.0) * lightS;
vColor = color * vec4(vec3(diffuse), 1.0) + ambientColor;
gl_Position = mvpMatrix * vec4(position, 1.0);
}
顶点着色器
precision mediump float;
varying vec4 vColor;
void main(void){
gl_FragColor = vColor;
}
片段着色器
好了,接下来我们的模型数据怎么和着色器进行数据链接呢,很简单,我们首先创建着色器的gl对象,用以js传参,请看代码
/**
* 生成着色器的函数
*/
function create_shader(id){
// 用来保存着色器的变量
var shader; // 根据id从HTML中获取指定的script标签
var scriptElement = document.getElementById(id); // 如果指定的script标签不存在,则返回
if(!scriptElement){return;} // 判断script标签的type属性
switch(scriptElement.type){ // 顶点着色器的时候
case 'x-shader/x-vertex':
shader = gl.createShader(gl.VERTEX_SHADER);
break; // 片段着色器的时候
case 'x-shader/x-fragment':
shader = gl.createShader(gl.FRAGMENT_SHADER);
break;
default :
return;
} // 将标签中的代码分配给生成的着色器
gl.shaderSource(shader, scriptElement.text); // 编译着色器
gl.compileShader(shader); // 判断一下着色器是否编译成功
if(gl.getShaderParameter(shader, gl.COMPILE_STATUS)){ // 编译成功,则返回着色器
return shader;
}else{ // 编译失败,弹出错误消息
alert(gl.getShaderInfoLog(shader));
}
}
这个函数生成的既可以是顶点着色器,也可以是片段着色器,在此不加赘述,有了着色器的gl对象,我们就能向着色器里传attribute和uniform参数了吗,显然不行,那么接下来我们就要构造一个可以向着色器对象传参数的程序对象gl.program,这也是难点之一,先看代码
/**
* 程序对象的生成和着色器连接的函数
*/
function create_program(vs, fs){
// 程序对象的生成
var program = gl.createProgram(); // 向程序对象里分配着色器
gl.attachShader(program, vs);
gl.attachShader(program, fs); // 将着色器连接
gl.linkProgram(program); // 判断着色器的连接是否成功
if(gl.getProgramParameter(program, gl.LINK_STATUS)){ // 成功的话,将程序对象设置为有效
gl.useProgram(program); // 返回程序对象
return program;
}else{ // 如果失败,弹出错误信息
alert(gl.getProgramInfoLog(program));
}
}
我们看到这个函数做了两件事,第一gl.attachShader将我们刚刚生成的着色器对象绑定到gl.program编程对象上,第二件事就是gl.useProgram激活绑定好着色器对象的编程对象,当然第二件事有待商榷,那就是如果我们有多个gl.program是不是每次创建绑定好着色器都要激活,这个要看具体使用场景,再次说明,这里这种绑定立即激活的方式不建议使用,一半都是绑定完成后等到要使用时才激活。为了这个还被lasoy老师批评了,哈哈,再次膜拜lasoy老师,阿里大佬。
好了,现在我们有了gl.program编程对象,就能够安心的向shader里传attribute和uniform参数了,具体传参方法不是我们这篇文章讨论的重点,请参考我的上一篇博客《原生WebGL场景中绘制多个圆锥圆柱》,链接地址https://www.cnblogs.com/ccentry/p/9864847.html。
接下来我们进入正题,持续绘制多个模型进一个canvas场景。也许同学们要说,这很简单啊,每次要绘制一个模型进入场景,就重复上述过程,先构造着色器对象gl.createShader(v-shader1),gl.createShader(f-shader1),然后绑定到程序对象gl.createProgram(program1)上,激活一下gl.useProgram(program1),接下来该穿attribute/uniform就传参,直接gl.drawElement()不就行了嘛,要绘制多少不同的模型就调用这个过程多少次,不就可以了嘛,哪来那么多废话,是不是。对于这种论调,我只能说,逻辑上是完全正确的,也能够正确无误地将多个长相各异的模型持续绘制进同一个canvas场景,没毛病。同学们就要喷了,那你bb了半天,想说啥呢?好,我就来说一说这么做的坏处是什么。请看下面场景
场景中的红色圆锥和红色圆柱的绘制方式就是类似刚才那种思想,不断重复构造着色器对象v-shader1 = gl.createShader(),f-shader1 = gl.createShader(),绑定编程program1 = gl.createProgram(v-shader1,f-shader1),激活编程对象gl.useProgram(program1),然后传attribute/uniform参数给着色器,空间位置姿态变换,gl.drawElement(),从而绘制出圆锥;再来就是重复这个过程绘制出圆柱,唯一稍有区别的是,我偷懒没重新构造shader对象,重新绑定program对象,而是重复利用同一套shader和program,只不过每次绘制传参attribute重新传了一次,覆盖前一次的attribute而已,原理其实一模一样,大家不要学我偷懒。有的同学看到结果以后更来劲了,你看看,这不是挺好吗,持续绘制多个不同模型成功了呀,有啥问题呀?那么我就说说问题在哪里。首先会发生交互的全局坐标系紊乱,请看下图
我们看到,整个模型错位了,原因就是空间变换矩阵并不能和每个模型相对于世界坐标系的相对局部坐标系矩阵正确相乘,这就是零散绘制多模型的坑。解决这个问题的方法就是采用树结构绘制子模型。这也是本文的核心论点,接下来我们就来看看如何采用树结构绘制,树的每个节点存储的又是什么对象。
由于osg和threejs都有自己的树结构,所以我也模仿二者自己构造了我的树,请看下面代码
/**
* 坐标系
* */
let Section = require('./Section');
let Operation = require('./Operation'); let Geometry = require('../core/Geometry');
let MatrixTransform = require('../core/MatrixTransform');
let StateSet = require('../core/StateSet');
let StateAttribute = require('../core/StateAttribute');
let BufferArray = require('../core/BufferArray');
let DrawElements = require('../core/DrawElements');
let Primitives = require('../core/Primitives');
let Depth = require('../core/Depth');
let LineWidth = require('../core/LineWidth');
let Material = require('../core/Material');
let BlendFunc = require('../core/BlendFunc');
let Algorithm = require('../util/Algorithm');
let BoundingBox = require('../util/BoundingBox');
let Vec3 = require('../util/Vec3');
let Vec4 = require('../util/Vec4');
let Plane = require('../util/Plane');
let Quat = require('../util/Quat');
let Mat4 = require('../util/Mat4');
let Utils = require('../util/Utils');
let ShaderFactory = require('../render/ShaderFactory');
let PolyhedronGeometry = require('../model/polyhedron');
let Group = require('../core/Group'); let CoordinateSection = function(viewer){
Section.call(this, viewer);
//坐标系模型的空间位置和姿态矩阵
this.position = Mat4.new();
this._coordRoot = undefined; this._scale = Vec3.create(, , );
this._translate = Vec3.new();
this._rotate = Quat.new(); this._scaleMatrix = Mat4.new();
this._translateMatrix = Mat4.new();
this._rotateMatrix = Mat4.new();
};
//继承Section类
CoordinateSection.prototype = Object.create(Section.prototype);
CoordinateSection.prototype.constructor = CoordinateSection; CoordinateSection.prototype = { /**
* 创建坐标系模型
* root:scene根节点
* */
create : function(root){
//初始化坐标系根节点
this._coordRoot = new MatrixTransform();
//几何
let polyhedronGeo = new PolyhedronGeometry();
//构造单位尺寸的模型
polyhedronGeo.getCone(0.2, 0.5, );
let geoms = polyhedronGeo.vertices;
let array = new Float32Array(geoms);
let vertexBuffer = new BufferArray(BufferArray.ARRAY_BUFFER, array, );
//面索引绘制方式
let indices = [];
indices = polyhedronGeo.faces;
//几何体类实例
let geom = new Geometry();
geom.setBufferArray('Vertex', vertexBuffer);
let index = new Int8Array(indices);
let indexBuffer = new BufferArray(BufferArray.ELEMENT_ARRAY_BUFFER, index, index.length);
let prim = new DrawElements(Primitives.TRIANGLES, indexBuffer);
geom.setPrimitive(prim);
//将几何对象加入坐标系根节点
this._coordRoot.addChild(geom);
//渲染组件
let stateSet = new StateSet();
//使用ColorDefaultProgram这组着色器
stateSet.addAttribute(ShaderFactory.createColorDefault.call(this));
stateSet.addAttribute(new Material([, 0.5, , ]));
stateSet.addAttribute(new BlendFunc(BlendFunc.SRC_ALPHA, BlendFunc.ONE_MINUS_SRC_ALPHA));
stateSet.addAttribute(new Depth(Depth.LESS, 0.1, 0.9, false));//深度值在中间
this._coordRoot.setStateSet(stateSet);
//将坐标系根节点加入场景根节点
root.addChild(this._coordRoot);
},
/**
* 调整坐标轴尺寸姿态
* boundingBox:scene场景的包围盒
* vec3Translate:场景平移向量
* vec4Rotate:场景旋转四元数
* vec3Scale:场景缩放向量
* mat4Scale:场景缩放矩阵
* mat4Translate:场景平移矩阵
* mat4Rotate:场景旋转矩阵
* worldMatrix:当前场景的世界坐标
* */
update: function (boundingBox, vec3Scale, vec3Translate, vec4Rotate, mat4Scale, mat4Translate, mat4Rotate, worldMatrix) {
if(boundingBox instanceof BoundingBox){//先保证boundingBox是BoundingBox类的实例
let vecSRaw = Vec3.new();
Vec3.copy(vecSRaw, vec3Scale);//克隆缩放向量,防止污染场景缩放向量
let vecS = Vec3.new();
this.computeScale(vecS, vecSRaw);//取场景缩放最长边的1/4作为坐标系模型缩放比例
let vecT = Vec3.new();
Vec3.copy(vecT, vec3Translate);//克隆平移向量,防止污染场景平移向量
let vecR = Vec4.new();
Vec4.copy(vecR, vec4Rotate);//克隆旋转向量,防止污染场景旋转向量
if (boundingBox.valid()) {//场景模型存在的话
let min = boundingBox.getMin();
let max = boundingBox.getMax();
boundingBox.getCenter(vec3Translate);
Vec3.sub(this._scale, max, min);
}
let matW = Mat4.new();
Mat4.copy(matW, worldMatrix); //克隆一个世界坐标系矩阵,防止修改场景包围盒的矩阵
let matS = Mat4.new();
Mat4.copy(matS, mat4Scale); //克隆一个缩放矩阵,防止污染场景包围盒的缩放矩阵
let matT = Mat4.new();
Mat4.copy(matT, mat4Translate); //克隆一个平移矩阵,防止污染场景包围盒的平移矩阵
let matR = Mat4.new();
Mat4.copy(matR, mat4Rotate); //克隆一个旋转矩阵,防止污染场景包围盒的旋转矩阵
Mat4.fromScaling(matS, vecS);
Mat4.fromTranslation(matT, vecT);
Mat4.fromQuat(matR, vecR); Mat4.mul(matW, matT, matR);
Mat4.mul(matW, matW, matS); this._coordRoot._matrix = matW;
}
},
//计算坐标系缩放比例
computeScale : function(newScale, boundingBoxScale){
//取场景模型包围盒最长一边的1/4
var scale = boundingBoxScale[] > boundingBoxScale[] ? boundingBoxScale[] : boundingBoxScale[];
scale = scale > boundingBoxScale[] ? scale : boundingBoxScale[];
scale *= /;
newScale[] = scale;
newScale[] = scale;
newScale[] = scale;
}
}; module.exports = CoordinateSection;
在这个构造类中,我将坐标系模型做成了一个根节点coordRoot,这个根节点下挂载了一个子模型(圆锥),该子模型下又挂载了三个子节点,一、geometry几何特征;二、transformMatrix千万注意是相对于他的父节点的空间变换矩阵,不是相对于世界坐标系的空间变换矩阵,千万注意;三、stateSet着色器相关对象,就是实现shader,program,传参attribute,uniform,空间变换,drawElement相关的配置和操作对象。这样做的好处就显而易见了,遍历整棵模型树,我既能将树上每一个节点都绑定不同的shader绘制出来,又能知道子节点相对于父节点的空间变换矩阵,就不会出现刚才那种错位的事了。
同学们看到这里应该明白树形结构加载多个子模型的好处了,由于这次的代码并不完整,osg也需要nodejs的运行环境,所以事先说明,贴出的代码只是为了帮助说明观点,本文代码只是局部关键部位,并不能运行,如有问题,可以交流。引用本文请注明出处https://www.cnblogs.com/ccentry/p/9903166.html
WebGL树形结构的模型渲染流程的更多相关文章
- 树形结构的数据渲染(element-ui&VUE)
在最开始学习的时候,渲染树形数据没有好好理解. 在实际的运用开发中,彻底的走了一遍树形数据,渲染角色权限的业务逻辑. 首先先发送请求获取全部权限树形结构, 其次发送请求获取当前用户的权限, 最后,通过 ...
- ejs模版实现递归树形结构渲染
使用过前端模板的同学们,尤其是使用过nodejs写后台服务的同学们,应该对ejs模板和jade模板都不陌生.对与ejs模板和jade模板孰强孰弱,载各大论坛中一直争论不休,有说ejs更直观的,也有说j ...
- 在NLP中深度学习模型何时需要树形结构?
在NLP中深度学习模型何时需要树形结构? 前段时间阅读了Jiwei Li等人[1]在EMNLP2015上发表的论文<When Are Tree Structures Necessary for ...
- 后台返回平铺数据,如何转换成树形json并渲染树形结构,ant tree 异步加载
如何后台返回对象数组(平铺式) 1.根据字段标识(板块)获取根节点 ### initTreeData(dataOrg){ var resultArr=dataOrg[0] var secArr=[]; ...
- 【Stage3D学习笔记续】山寨Starling(三):Starling核心渲染流程
这篇文章我们剔除Starling的Touch事件体系和动画体系,专门来看看Starling中的渲染流程实现,以及其搭建的显示列表结构. 由于Starling是模仿Flash的原生显示列表,所以我们可以 ...
- Ogre内部渲染流程分析系列
come from:http://blog.csdn.net/weiqubo/article/details/6956005 要理解OGRE引擎,就要理解其中占很重要位置的 Renderable接口, ...
- 转:Ogre内部渲染流程
以下是 Ogre 的代码中的详细说明: Renderable是OGRE中所有可渲染对象的抽象接口 这个接口抽象出了在渲染管线中的被分组的离散的可渲染对象基本的方法. 此接口的实现类必须是基于单一的材质 ...
- [Swing]树形结构的实现
一般步骤: 1.建立根节点 private DefaultMutableTreeNode root = new DefaultMutableTreeNode("根节点"); 2.建 ...
- 浏览器渲染流程&Composite(渲染层合并)简单总结
梳理浏览器渲染流程 首先简单了解一下浏览器请求.加载.渲染一个页面的大致过程: DNS 查询 TCP 连接 HTTP 请求即响应 服务器响应 客户端渲染 这里主要将客户端渲染展开梳理一下,从浏览器器内 ...
随机推荐
- ajax传递数组,后台接收为null解决方法
traditional:true,加上这个就好,默认为false,即允许深度序列化参数,但是servlet api不支持,所有设为true阻止就好了. $.ajax({ type:'post', ur ...
- Python KafkaProducer and KafkaConsumer的开发模块
1.在python中往kakfa写数据和读取数据,使用的是python-kafka库 2.消费者需持续写入数据,因groupid存在偏移量,才能看看到数据. 3.安装库的命令为pip install ...
- DPDK+Pktgen 高速发包测试
参考博客 Pktgen概述 Pktgen,(Packet Gen-erator)是一个基于DPDK的软件框架,发包速率可达线速. 提供运行时管理,端口实时测量. 可以控制 UDP, TCP, ARP, ...
- C#简单实现LRU缓存
最近跟同学吃饭扯淡的时候,由技术扯到薪资,又由薪资扯到他找工作时跟面试官是怎么扯淡拿高工资的,各种技术一顿侃,总之只要啥都了解就没问题了.谈到缓存的时候,我试探性的问了问- -你还记得LRU怎么写吗, ...
- 【CSS-移动端响应式布局详解】
背景 移动端响应式布局开发主要方案有: 基于rem开发 基于媒体查询 基于弹性盒 基础概念 在讨论响应式布局知识前,先了解下移动端常用基础概念. 逻辑像素(CSS pixels) 浏览器使用的抽象单位 ...
- Linux-- 查看文件 more与其它
more 翻页查看 用法:more 文件名 nl 显示行号打印(不常用) 1.不打印空白行行号:nl -b t 文件名 类似 cat -b 文件名 2.打印所有行行号:nl -b a 文件名 类似 c ...
- 使用属性Props完成一张卡片
一:我们先安装bootstrap,为了使我们的样式好看些 cnpm install bootstrap --save 二:我们在index.js中引入bootstap Import ‘bootst ...
- MySQL——安装
1. 下载源: http://repo.mysql.com/yum/mysql-8.0-community/el/7/x86_64/mysql80-community-release-el7-2.no ...
- shiro使用框架,自定义过滤器
1.shiro配置文件配置 <!-- Shiro Filter --> <bean id="shiroFilter" class="org.apache ...
- mysql 8 windows 版本zip方式安装步骤
mysql 8 windows 版本zip方式安装步骤(下载地址:https://dev.mysql.com/downloads/mysql/)1,解压ZIP文件到指定目录下:如D:\mysql-8. ...