大家应该知道3D世界中任何的面都是由三角形绘制完成的,因为任何无规则的集合图形都可以由三角形来组成。比如四边形,无论是正四边形还是无规则四边形都可以由两个三角形拼接而成。结合本文的标题大家仔细想想,如果需要绘制一个动态无规则面其实只需要得到动态的两个轨迹点即可,那么结合下面的图片大家仔细在想想。

        暂时我们先忽略Z轴(这样在平面中看得更清楚),假设Z轴坐标都为0。假设游戏中有两个轨迹点在动态的增加与改变,最后将这两个点改变的轨迹拼接起来就是它们生成的面。如上图所示,第一个点的轨迹是“ 3,4,5,6,7” 第二个点的轨迹是“2,1,10,9,8” 。这两个点的长度是可变的,前提是他们两个的数量必需完全一样。接着,如下图所示,我们将这些点两两相连起来,目前一共形成了8个三角形面(可根据两个动态点的数量而确定整个网格面三角形面的数量)。最后我们将这8个三角形填充上同样的颜色,就可以实现一个完整的立体网格面。

         原理很简单,就是这样的我相信大家看到这里大家都能明白,接着我们就学习如何使用代码来实现它。首先创建Unity工程,接着创建一个空的游戏对象,然后给该游戏对象绑定Mesh Filter组件 与 Mesh Renderer组件。

Mesh Filter组件:表示网格面,这个网格面是由我们使用代码将所有三角形拼接起来生成的面。

Mesh Renderer组件:表示表示网格的渲染,可设置一个渲染的材质,它包括贴图与颜色。

       如下图所示,我说说里面比较重要的属性。Mesh Renderer中,Materials下拉列表中可设置网格模型的材质,此时我们设置了一个红色的材质。 Mesh Filter:目前为None,也不用再编辑器中为它赋值,因为这个网格模型我们会在代码中生成并且赋值。在下面就是方刚我们设置红色的材质资源,Shader中设置了贴图的属性,目前是GUI/ TextShader。它表示这个材质的渲染级别在GUI上,就是优先级是最一层的。举个例子无论在这个网格模型的前面绘制多少模型,它永远都会在最前面显示。就这个例子而言它的存在并不是必需的,其实Shader的选项还有很多,可透明、不可透明、镜面、反射等等,后期我会向大家详细道来。

OK,现在资源文件都已经准备完毕,下面我们学习如何来绘制一个三角形,从简单的开始。。把下面的代码绑定在摄像机对象当中。

  1. void Start()
  2. {
  3. //得到MeshFilter对象,目前是空的。
  4. MeshFilter meshFilter = (MeshFilter)GameObject.Find("face").GetComponent(typeof(MeshFilter));
  5. //得到对应的网格对象
  6. Mesh mesh = meshFilter.mesh;
  7.  
  8. //三角形顶点的坐标数组
  9. Vector3[] vertices = ];
  10. //三角形顶点ID数组
  11. ];
  12.  
  13. //三角形三个定点坐标,为了显示清楚忽略Z轴
  14. vertices[] = , , );
  15. vertices[] = , , );
  16. vertices[] = , , );
  17.  
  18. //三角形绘制顶点的数组
  19. triangles[] = ;
  20. triangles[] = ;
  21. triangles[] = ;
  22.  
  23. mesh.vertices = vertices;
  24. mesh.triangles = triangles;
  25.  
  26. }

          代码中有两个非常重要的概念,就是三角形顶点数组与坐标数组。先说说坐标数组,假设需要绘制一个四边形,此时三角形坐标数组的长度应当是4,它保存着四边形四个顶点的坐标。然后是顶点数组,四边形是由两个三角形组成,然而一个三角形是由3个顶点组成,两个三角形就应当是6个顶点组成,无论多少个三角形它们的结构都应当是以此类推。

注解1:这里是将模型的顶点数组与坐标数组赋值给网格模型,还记得刚刚在创建Mesh Filter时,当时没有在编辑器中给网格模型赋值,实际上代码走到这里就会重新为网格模型MeshFilter赋值,接着我们在代码中绘制的三角形就会显示在屏幕当中。

如图所示,三角形已经绘制在屏幕当中。

图中数组 0 1 2 表示该三角形的三个顶点的ID。这个ID对应代码中对应vertices数组索引顶点的坐标。也就是说(0,0,0)对应顶点0,(0,1,0)对应顶点1,(1,0,0)对应顶点2

下面我们修改一下代码,让屏幕中一共绘制4个三角形。

  1. //网格模型顶点数量
  2. ;
  3. void Start()
  4. {
  5. //得到MeshFilter对象,目前是空的。
  6. MeshFilter meshFilter = (MeshFilter)GameObject.Find("face").GetComponent(typeof(MeshFilter));
  7. //得到对应的网格对象
  8. Mesh mesh = meshFilter.mesh;
  9.  
  10. //三角形顶点的坐标数组
  11. Vector3[] vertices = new Vector3[VERTICES_COUNT];
  12.  
  13. //得到三角形的数量
  14. ;
  15.  
  16. //三角形顶点ID数组
  17. ];
  18.  
  19. //三角形三个定点坐标,为了显示清楚忽略Z轴
  20. vertices[] = , , );
  21. vertices[] = , , );
  22. vertices[] = , , );
  23. vertices[] = , , );
  24. vertices[] = , , );
  25. vertices[] = , , );
  26.  
  27. //三角形绘制顶点的数组
  28. triangles[] = ;
  29. triangles[] = ;
  30. triangles[] = ;
  31.  
  32. triangles[] = ;
  33. triangles[] = ;
  34. triangles[] = ;
  35.  
  36. triangles[] = ;
  37. triangles[] = ;
  38. triangles[] = ;
  39.  
  40. triangles[] = ;
  41. triangles[] = ;
  42. triangles[] = ;
  43.  
  44. //绘制三角形
  45. mesh.vertices = vertices;
  46. mesh.triangles = triangles;
  47. }

         已知模型的顶点数量,顶点数量减去2就是三角形的数量,三角形的数量在乘以3就是三角形顶点的数量。根据这个公式计算得知,上述代码中共绘制4个三角形,顶点坐标数组应当是6,顶点ID数组应当是12。多个三角形在顶点ID数组中排列方式比较特殊,大家需要仔细记录一下不然无法绘制出正确的三角形。如下图所示,由于我这边没有合适的3D坐标点,就用正三角形拼接出一个正四边形,这个四边形是由6个顶点4个小三角形组成 ,看到这里思路清晰的朋友应当明了无规则四边形的绘制原理和它完全一样。只需要传入适当的3D坐标点即可。

          根据上面的逻辑,我们修改一下算法。假设三角形的顶点坐标为任意数量,我们需要更根据顶点坐标数量来计算对应顶点ID的数组内容。在for循环中start =0 与end =3的含义是绘制从顶点坐标数组中索引为0的顶点开始绘制到数组索引为3的顶点,也就说是这里从0到3绘制了3个三角形。

  1. //网格模型顶点数量
  2. ;
  3. void Start()
  4. {
  5. //得到MeshFilter对象,目前是空的。
  6. MeshFilter meshFilter = (MeshFilter)GameObject.Find("face").GetComponent(typeof(MeshFilter));
  7. //得到对应的网格对象
  8. Mesh mesh = meshFilter.mesh;
  9. //三角形顶点的坐标数组
  10. Vector3[] vertices = new Vector3[VERTICES_COUNT];
  11. //得到三角形的数量
  12. ;
  13. //三角形顶点ID数组
  14. ];
  15. //三角形三个定点坐标,为了显示清楚忽略Z轴
  16. vertices[] = , , );
  17. vertices[] = , , );
  18. vertices[] = , , );
  19. vertices[] = , , );
  20. vertices[] = , , );
  21. vertices[] = , , );
  22.  
  23. //绘制三角形
  24. mesh.vertices = vertices;
  25. //起始三角形顶点
  26. ;
  27. //结束三角形的顶点
  28. ;
  29. for (int i = start; i < end; i++)
  30. {
  31. ; j < ; j++)
  32. {
  33. == )
  34. {
  35. triangles[ * i + j] = i + j;
  36. }
  37. else
  38. {
  39. triangles[ * i + j] = i + - j;
  40. }
  41. }
  42. }
  43. mesh.triangles = triangles;
  44. }

      如下图所示,根据上面的逻辑算法,共绘制了3个三角形,并且顶点坐标ID是由 0 到3 。 说到这里请大家仔细想想本文的标题内容,其实两个动态轨迹的点就是在维护triangles顶点坐标数组。triangles[0]、triangles[2]、triangles[4]……表示一个轨迹点的值,triangles[1]、triangles[3]、triangles[5]……就表示另一个轨迹点的值,最终将它们通过上面的算法将三角形面连接起来那么就是动态的两个点轨迹绘制面了。

         Unity3D其实非常好玩,上手虽然很简单,但是想深入其实并没有那么容易,今天这篇文章的思路已经写完,如果还是没能明白的朋友请仔细揣摩三角形与四边形之间的却别,哇咔咔,已经不早了我也得睡觉了,明天还得上班 GOGOGOGOGO加油~希望我们大家可以共同学习,共同进步。

【转载】Unity3D研究院之与根据动态的两个轨迹点绘制面详解的更多相关文章

  1. 技巧:Linux 动态库与静态库制作及使用详解

    技巧:Linux 动态库与静态库制作及使用详解 标准库的三种连接方式及静态库制作与使用方法 Linux 应用开发通常要考虑三个问题,即:1)在 Linux 应用程序开发过程中遇到过标准库链接在不同 L ...

  2. 转载文章-----Rational Rose2007(v7.0)下载地址、安装及激活详解教程(图)

    转载地址:http://www.cnblogs.com/leaven/p/3718361.html 最近需要画uml图,之前用的是Rose 2003版的,由于好久没进去了,结果发现原来的激活又失效了, ...

  3. (转载)Unity3D研究院之使用 C#合成解析XML与JSON(四十一)

    XML与JSON在开发中非常重要, 其实核心就是处理字符串.一个是XML的字符串一个是JSON的字符串,尤其是在处理网络请求的时候,肯定是要用的.另外现在JSON非常的流行,我写了一个简单的例子融合了 ...

  4. Unity3D研究院之游戏对象的访问绘制线与绘制面详解(十七)

    一眨眼学习Unity3D 也有一段时间了,基本已经拿下了这套游戏引擎,回过头来想想以前写的RPG 游戏引擎,越来越发现以前写的就是垃圾.人果然是要不断学习与不断进步,好好学习,天天向上.哇咔咔- 加油 ...

  5. 转载:MySQL数据库INSERT、UPDATE、DELETE以及REPLACE语句的用法详解

    转自:http://www.jb51.net/article/39199.htm 本篇文章是对MySQL数据库INSERT.UPDATE.DELETE以及REPLACE语句的用法进行了详细的分析介绍, ...

  6. 动态SQL各个标签作用以及注意事项详解

    创建com.mybatis包,包含:UserMapper.xml和mybatis-config.xml UserMapper.xml代码: <?xml version="1.0&quo ...

  7. 转载:MySQL EXPLAIN 命令详解学习

    转载自:https://blog.csdn.net/mchdba/article/details/9190771 MySQL EXPLAIN 命令详解 MySQL的EXPLAIN命令用于SQL语句的查 ...

  8. [转载]详解ssh端口转发(二)

    关于使用ssh portforwarding来进行FQ的操作,网络上已经有很多很好的文章,我在这里只是画两个图解释一下. 首先要记住一件事情就是: SSH 端口转发自然需要 SSH 连接,而 SSH ...

  9. (转载)实例详解Android快速开发工具类总结

    实例详解Android快速开发工具类总结 作者:LiJinlun 字体:[增加 减小] 类型:转载 时间:2016-01-24我要评论 这篇文章主要介绍了实例详解Android快速开发工具类总结的相关 ...

随机推荐

  1. iOS逆向命令集

    越狱命令行 破壳: 10.10.215.119 ssh root@10.10.215.119 ssh root@10.10.213.176 CCBMobileBank Fuqianlade-iPhon ...

  2. Gym 100169A 最短路

    题意:不久后滑铁卢将会变得非常冷,但是幸运的是,很多建筑都被桥梁和隧道连接着,所以你不需要总是走在外面.但是现在建筑 物之间的连接是错综复杂的,很难知道某两个建筑物之间的最优路线,所以需要你写程序判断 ...

  3. 数黑格有多少个,模拟题,POJ(1656)

    题目链接:http://poj.org/problem?id=1656 #include <stdio.h> #include <iostream> #include < ...

  4. 表面积最小(POJ3536)

    题目链接:http://poj.org/problem?id=3536 在体积固定的情况下,表面积最小时的长,宽,高. 这里枚举长,宽,根据体积计算高. #include <iostream&g ...

  5. linux .h .so .a文件

    在linux开发中,完全不使用第三方库的情况比较少见,通常都需要借助一个或多个函数库的支持才能完成相应功能.从程序员角度看,函数库实际上是一些头文件(.h)和库文件(.so或.a)的集合.linux下 ...

  6. R 语言学习日志 1

      1. CSV文件的的读取与写出 2. 数据集筛选 3. 简单随机抽样 sample函数   正文: 1. CSV文件的的读取与写出 文件读取: df2 <- read.table(" ...

  7. redis事务中的WATCH命令和基于CAS的乐观锁

    转自:http://blog.sina.com.cn/s/blog_ae8441630101cgy3.html 在Redis的事务中,WATCH命令可用于提供CAS(check-and-set)功能. ...

  8. 开源项目托管github步骤

    一.在github新建项目,复制到本地更改之后命令提交. 1.进入github主页新建项目:https://github.com/ccyinghua 2.复制项目地址 3.打开git Bash 命令行 ...

  9. Python 初始—(装饰器)

    本质上也是函数的另一种表现形式,为其它函数丰富其功能,装饰其他函数,附加功能 在不改变被装饰的函数的源代码,而且不改变被装饰的函数的调用方式,因此被装饰的函数感知不到装饰器函数的存在 分解装饰器 函数 ...

  10. Web as a App(Web既APP)的概念可以提出吗?

    Web as a App (WaaA),Web既APP.灵感出于SaaS.PaaS.IaaS等~ 最近在做一个PC项目,即便我的项目是to B的,用户量没那么大,但是我仍然很注重性能及用户体验,我把我 ...