当前位置: 首页 > wzjs >正文

广东微信网站制作公司做网站 信息集成过程的顺序

广东微信网站制作公司,做网站 信息集成过程的顺序,东莞短视频推广哪个平台好,北京网站设计公司招聘信息层次结构模型 三维模型和现实中的人类或机器人不一样,它的部件并没有真正连接在一起。如果直接转动上臂,那么肘部以下的部分,包括前臂、手掌和手指,只会留在原地,这样手臂就断开了。 所以,当上臂绕肩关节转…

层次结构模型

三维模型和现实中的人类或机器人不一样,它的部件并没有真正连接在一起。如果直接转动上臂,那么肘部以下的部分,包括前臂、手掌和手指,只会留在原地,这样手臂就断开了。

所以,当上臂绕肩关节转动时,需要在代码中实现“肘部以下部分跟随上臂转动“的逻辑。具体地,上臂绕肩关节转动了多少度,肘部以下的部分也应该绕肩关节转动多少度。

简单情况

实现“部件A转动带动部件B转动”可以很直接,只要对部件B也施以部件A的旋转矩阵即可。比如,使用模型矩阵使上臂绕肩关节转动30度,然后在绘制肘关节以下的各部位时,为它们施加同一个模型矩阵,也令其绕肩关节转动30度,这样,肘关节以下的部分就能自动跟随上臂转动了。

复杂情况

比如先使上臂绕肩关节转动30度,然后使前臂绕肘关节转动10度,那么对肘关节以下的部分,你就得先施加上臂绕肩关节转动30度的矩阵(可称为“肩关节模型矩阵”),然后再施加前臂绕肘关节转动10度的矩阵。将这两个矩阵相乘,其结果可称为“肘关节模型矩阵”,那么在绘制肘关节以下部分的时候,直接应用这个所谓的“肘关节模型矩阵”(而不考虑肩关节,因为肩关节的转动信息已经包含在该矩阵中了)作为模型矩阵就可以了。

单关节模型

单关节完整代码案例在这:singleNode.html

绘制长方体作为上臂

还是采用drawElement的方法,绘制长方体,和之前绘制正方体的绘制方式一样。只是调整了一个顶点坐标的值。

function initBuffer() {var vertices = new Float32Array([1.5, 10.0, 1.5, -1.5, 10.0, 1.5, -1.5, 0.0, 1.5, 1.5, 0.0, 1.5, // v0-v1-v2-v3 front1.5, 10.0, 1.5, 1.5, 0.0, 1.5, 1.5, 0.0, -1.5, 1.5, 10.0, -1.5, // v0-v3-v4-v5 right1.5, 10.0, 1.5, 1.5, 10.0, -1.5, -1.5, 10.0, -1.5, -1.5, 10.0, 1.5, // v0-v5-v6-v1 up-1.5, 10.0, 1.5, -1.5, 10.0, -1.5, -1.5, 0.0, -1.5, -1.5, 0.0, 1.5, // v1-v6-v7-v2 left-1.5, 0.0, -1.5, 1.5, 0.0, -1.5, 1.5, 0.0, 1.5, -1.5, 0.0, 1.5, // v7-v4-v3-v2 down1.5, 0.0, -1.5, -1.5, 0.0, -1.5, -1.5, 10.0, -1.5, 1.5, 10.0, -1.5  // v4-v7-v6-v5 back]);// Normalvar normals = new Float32Array([0.0, 0.0, 1.0, 0.0, 0.0, 1.0, 0.0, 0.0, 1.0, 0.0, 0.0, 1.0, // v0-v1-v2-v3 front1.0, 0.0, 0.0, 1.0, 0.0, 0.0, 1.0, 0.0, 0.0, 1.0, 0.0, 0.0, // v0-v3-v4-v5 right0.0, 1.0, 0.0, 0.0, 1.0, 0.0, 0.0, 1.0, 0.0, 0.0, 1.0, 0.0, // v0-v5-v6-v1 up-1.0, 0.0, 0.0, -1.0, 0.0, 0.0, -1.0, 0.0, 0.0, -1.0, 0.0, 0.0, // v1-v6-v7-v2 left0.0, -1.0, 0.0, 0.0, -1.0, 0.0, 0.0, -1.0, 0.0, 0.0, -1.0, 0.0, // v7-v4-v3-v2 down0.0, 0.0, -1.0, 0.0, 0.0, -1.0, 0.0, 0.0, -1.0, 0.0, 0.0, -1.0  // v4-v7-v6-v5 back]);// Indices of the verticesindices = new Uint8Array([0, 1, 2, 0, 2, 3,    // front4, 5, 6, 4, 6, 7,    // right8, 9, 10, 8, 10, 11,    // up12, 13, 14, 12, 14, 15,    // left16, 17, 18, 16, 18, 19,    // down20, 21, 22, 20, 22, 23     // back]);let pointPosition = new Float32Array(vertices);let aPsotion = webgl.getAttribLocation(webgl.program, 'a_position');let triangleBuffer = webgl.createBuffer();webgl.bindBuffer(webgl.ARRAY_BUFFER, triangleBuffer);webgl.bufferData(webgl.ARRAY_BUFFER, pointPosition, webgl.STATIC_DRAW);webgl.enableVertexAttribArray(aPsotion);webgl.vertexAttribPointer(aPsotion, 3, webgl.FLOAT, false, 0, 0);let aNormal = webgl.getAttribLocation(webgl.program, 'a_Normal');let normalsBuffer = webgl.createBuffer();let normalsArr = new Float32Array(normals);webgl.bindBuffer(webgl.ARRAY_BUFFER, normalsBuffer);webgl.bufferData(webgl.ARRAY_BUFFER, normalsArr, webgl.STATIC_DRAW);webgl.enableVertexAttribArray(aNormal);webgl.vertexAttribPointer(aNormal, 3, webgl.FLOAT, false, 0, 0);let indexBuffer = webgl.createBuffer();let indices1 = new Uint8Array(indices);webgl.bindBuffer(webgl.ELEMENT_ARRAY_BUFFER, indexBuffer);webgl.bufferData(webgl.ELEMENT_ARRAY_BUFFER, indices1, webgl.STATIC_DRAW);
}

添加光照

直接添加光照的代码是放在initBuffer当中,现在将其抽离成一个单独的函数,数据都是传递给着色器的,不影响绘制。

function initLight() {let u_DiffuseLight = webGL.getUniformLocation(program, 'u_DiffuseLight');webGL.uniform3f(u_DiffuseLight, 1.0, 1.0, 1.0);let u_LightDirection = webGL.getUniformLocation(program, 'u_PointLightPosition');webGL.uniform3fv(u_LightDirection, [3.0, 3.0, 4.0]);let u_AmbientLight = webGL.getUniformLocation(program, 'u_AmbientLight');webGL.uniform3f(u_AmbientLight, 0.8, 0.8, 0.8);
}

设置矩阵

现在已经准备好了绘制的数据,之后就是通过设置透视投影矩阵和模型矩阵来绘制了。也单独抽离成一个方法。

那么在这里返回了一个模型矩阵的目的是,第一次绘制长方体时是上臂,这个时候调用initTransformation会拿到上臂的模型矩阵,那么通过上臂再去绘制小臂的时候用大臂的模型矩阵再作为小臂变换的顶点矩阵,也就完成了大臂和小臂的联动

  function initTransformation(angele, rotateArr, ModelMatrix = mat4.create()) {let ProjMatrix = mat4.create();mat4.identity(ProjMatrix);mat4.perspective(ProjMatrix, angle * Math.PI / 180, webGLdiv.clientWidth / webGLdiv.clientHeight, 1, 1000); //修改可视域范围let uniformMatrix1 = webGL.getUniformLocation(program, 'u_formMatrix');mat4.rotate(ModelMatrix, ModelMatrix, (angele * Math.PI) / 180.0, rotateArr);let ViewMatrix = mat4.create();mat4.identity(ViewMatrix);mat4.lookAt(ViewMatrix, [50, 50, 50], [0, 0, 0], [0, 1, 0]);let mvMatrix = mat4.create();mat4.identity(mvMatrix);mat4.multiply(mvMatrix, ViewMatrix, ModelMatrix);let mvpMatrix = mat4.create();mat4.identity(mvpMatrix);mat4.multiply(mvpMatrix, ProjMatrix, mvMatrix);webGL.uniformMatrix4fv(uniformMatrix1, false, mvpMatrix);return ModelMatrix;
}

绘制

这里就是上一步所说的联动,先绘制上臂,再绘制小臂,小臂的模型矩阵是上臂的模型矩阵。

function draw() {let modelArr = initTransformation(jointAngle, [0, 1, 0]);webGL.drawElements(webGL.TRIANGLES, indices.length, webGL.UNSIGNED_BYTE, 0);initTransformation(armAngle, [0, 0, 1], modelArr);webGL.drawElements(webGL.TRIANGLES, indices.length, webGL.UNSIGNED_BYTE, 0);
}function clear() {webGL.clearColor(0, 0, 0, 1);webGL.clear(webGL.COLOR_BUFFER_BIT | webGL.DEPTH_BUFFER_BIT);webGL.enable(webGL.DEPTH_TEST);
}

添加键盘控制事件

添加键盘事件控制jointAngle、armAngle来控制大臂和小臂的旋转角度。

function initEvent() {document.onkeydown = keydown;
}function keydown(ev) {switch (ev.keyCode) {case 38:if (jointAngle < 135.0) jointAngle += ANGLE_STEP;break;case 40:if (jointAngle > -135.0) jointAngle -= ANGLE_STEP;break;case 39:armAngle += ANGLE_STEP;break;case 37:armAngle -= ANGLE_STEP;break;default:return;}clear();draw();
}

那么最后的效果就是这样啦

在这里插入图片描述

多节点模型

那么上一步完成了大臂和小臂的联动,如果现在需要绘制一个简单的有头有手的机器人模型,下面简单的说一下实现步骤:

  • 先绘制头部,用头部的模型矩阵来绘制上半身,完成头部、上半身联动。
  • 用上半身的模型矩阵绘制左大臂,再通过大臂的模型矩阵来绘制小臂,
  • 再通过小臂的模型矩阵来绘制手指1和手指2,这样就完成了大臂、小臂、手指的联动。

不管他有多少个节点,只要有节点的模型矩阵,就可以通过这个模型矩阵来绘制节点,完成联动。

着色器对象 initShader

在前面已经简单解释了一下initShader是用来干啥的,这个后续都没有做过修改。现在来深入探究一下。

initShader函数的作用是:编译GLSLES代码,创建和初始化着色器供WebGL使用。具体地,分为以下7个步骤:

  • 创建着色器对象(gl.createShader())
  • 向着色器对象中填充着色器程序的源代码(gl.shaderSource())
  • 编译着色器(gl.compileShader())
  • 创建程序对象(gl.createProgram())
  • 为程序对象分配着色器(gl.attachShader())
  • 连接程序对象(gl.linkProgram())
  • 使用程序对象(gl.useProgram())

这里出现了两个对象:着色器对象、程序对象

  • 着色器对象:着色器对象管理一个顶点着色器或一个片元着色器。每一个着色器都有一个着色器对象
  • 程序对象:程序对象是管理着色器对象的容器。WebGL中,一个程序对象必须包含一个顶点着色器和一个片元着色器

创建着色器对象

所有的着色器对象都是以gl.createShader()创建的,这个函数接收一个参数,这个参数是gl.VERTEX_SHADER或gl.FRAGMENT_SHADER,分别表示顶点着色器和片元着色器。

如果不需要这个着色器,可以通过gl.deleteShader()删除这个着色器对象。

指定着色器代码

通过gl.shaderSource()指定着色器的源代码,这个函数接收两个参数,第一个参数是着色器对象,第二个参数是着色器的源代码。

编译着色器

GLSL
ES语言和JavaScript不同而更接近C或C++,在使用之前需要编译成二进制的可执行格式,WebGL系统真正使用的是这种可执行格式。使用gl.compileShader()
函数进行编译。

当对着色器编译之后,如果编译失败,可以通过gl.getShaderParameter()
函数获取着色器的编译状态,如果编译失败,可以通过gl.getShaderInfoLog()函数获取着色器的编译信息。

if (!webGL.getShaderParameter(vsShader, webGL.COMPILE_STATUS)) {console.log('vsShader error =====', webGL.getShaderInfoLog(vsShader));return;
}
if (!webGL.getShaderParameter(fsShader, webGL.COMPILE_STATUS)) {console.log('fsShader error =====', webGL.getShaderInfoLog(fsShader));return;
}

创建程序对象

调用gl.createProgram()创建程序对象,这个函数返回一个程序对象。类似的,可以通过gl.deleteProgram()
删除程序对象。一旦程序对象被创建之后,需要向程序附上两个着色器

为程序对象分配着色器

WebGL系统要运行起来,必须要有两个着色器:一个顶点着色器和一个片元着色器。可以使用gl.attachShader()函数为程序对象分配这两个着色器。

着色器在附给程序对象前,并不一定要为其指定代码或进行编译(也就是说,把空的着色器附给程序对象也是可以的)。类似地,可以使用gl.detachShader()
函数来解除分配给程序对象的着色器。

连接程序对象

在为程序对象分配了两个着色器对象后,还需要将(顶点着色器和片元)着色器连接起来。使用gl.1inkProgram()函数来进行这一步操作。

程序对象进行着色器连接操作,目的是保证:

  • 顶点着色器和片元着色器的varying变量同名同类型,且一一对应
  • 顶点着色器对每个varying变量赋了值
  • 顶点着色器和片元着色器中的同名uniform变量也是同类型的(无需一一对应,即某些uniform变量可以出现在一个着色器中而不出现在另一个中)
  • 着色器中的attribute变量、uniform变量和varying变量的个数没有超过着色器的上限

使用程序对象

通过调用gl.useProgram()告知WebGL系统绘制时使用哪个程序对象


文章转载自:

http://7szWJHwg.yfstt.cn
http://SKnP58QC.yfstt.cn
http://RAjH50cX.yfstt.cn
http://jAwLgab9.yfstt.cn
http://TAd5sxbc.yfstt.cn
http://kSO5Rn70.yfstt.cn
http://7gb3I5lv.yfstt.cn
http://nyIMupop.yfstt.cn
http://XqZAEPVo.yfstt.cn
http://HueulqCR.yfstt.cn
http://be2Qpc1X.yfstt.cn
http://s1BWyIWr.yfstt.cn
http://xIfqynzo.yfstt.cn
http://QexsbSaS.yfstt.cn
http://XTeKMyhE.yfstt.cn
http://H9B5BXve.yfstt.cn
http://Ipu1Qz6b.yfstt.cn
http://ouaUhCkA.yfstt.cn
http://DJjLdpmS.yfstt.cn
http://vp7Q09vG.yfstt.cn
http://q6AqJQZ8.yfstt.cn
http://UdOl7w3D.yfstt.cn
http://Y7mur80e.yfstt.cn
http://be53sjE3.yfstt.cn
http://MKjFqiHW.yfstt.cn
http://ZTS1aCXy.yfstt.cn
http://WXgiN7G2.yfstt.cn
http://mtQ1XvUK.yfstt.cn
http://ZvmPpwWc.yfstt.cn
http://d0H63dMk.yfstt.cn
http://www.dtcms.com/wzjs/774200.html

相关文章:

  • 创建网站的六个步骤 天堂最新版在线资源
  • 给别人做网站的话术网站怎么做的qq邮件订阅
  • 网站展示济南网站制作搜到
  • 广州市律师网站建设怎么样百度手机助手应用商店
  • 高密 网站建设西安网站开发技术
  • 网站建设流程图viso网站开发移动端多少钱
  • 网站被抄袭怎么投诉肥乡邯郸做网站
  • 实用软件推荐wordpress头部优化
  • php网站支付宝接口响应式网页源码
  • 建设网站涉及的技术佛山网页设计公司
  • wordpress企业网站seowordpress 4.7 新功能
  • 网站建设平台开发网站建设公司 深圳
  • 建站平台哪家好郑州建设网站设计
  • 美食网站网页设计代码企业整体vi设计
  • 效果好的免费网站建设手机wordpress建站教程
  • 做网站贵么孝感市建设局网站
  • 上海企业都用什么网站国内 上市网站建设公司排名
  • 新洲建设投标网站高校网站建设与管理问题分析
  • 如何看一个网站的备案在哪里做的哪家网站做推广好
  • 外贸网站营销建站高德地图可以看国外的地图吗
  • 做家装的网站有哪些内容做网站用啥软件
  • 怎么把一个网站的关键词茶楼网站源码
  • 海口建设网站的公司好一点的网站建设公司
  • 做网站的linux程序代码营销推广网站
  • 设计师网上接单的网站企业网页界面设计
  • 做网站收入来源表wordpress 文章查看次数
  • 网站建设商城建设北京哪里招聘网页设计
  • vip视频解析网站怎么做徐州品牌网站建设|徐州网站优化|徐州网络公司-徐州启思信息科技
  • seo网站三要素怎么做如何查看自己的企业邮箱
  • 做盗版视频网站吗北京做网站优化多少钱