关于Unity中Mesh网格的详解

3D模型

通过3D建模软件所建出来的点和面,如以三角形为主的点和面,比如人的脑袋一个球,就是由各种各样的三角形组成的点和面。

点和面以及纹理坐标都是通过3D建模软件建模出来的。

Unity会帮我们把模型的信息存到Mesh里面来,Mesh翻译成中文是网格。

顶点,三角形,纹理坐标,法线和切线。

 

 

3D建模软件

1:Autodesk 3D Studio Max 支持mac os windows;
2: Autodesk 3D Maya 支持windows
3: Cinema4D 支持mac os windows
4: Blender 开源跨平台的全能三维制作软件, 支持mac os windows, linux;
5: Cheetah3D: 支持mac os
6: Unity与建模软件的单位比例:
unity系统单位为m, 建模软件的m的尺寸大小不一样,所以导入的时候有差异:

        内部米      导入unity后的尺寸/m        与Unity单位的比例关系
3Dmax         1        0.01              100:1
Maya            1        100               1:100
Cinema 4D       1        100               1:100
Light Wave       1        0.01                100:1

 


网格Mesh

1: Unity提供一个Mesh类,允许脚本来创建和修改,通过Mesh类能生成或修改物体的网格,能做出非常酷炫的物体变形特效;
2: Mesh filter 网格过滤器从资源中拿出网格并将其传递给MeshRender,用于绘制, 导入模型的时候,Unity会自动创建一个这样的组件;
3: Mesh 是网格过滤器实例化的Mesh, Mesh中存储物体的网格数据的属性和生成或修改物体网格的方法
4: 点---->顶点数组<Vector3>: 每个顶点的x, y, z坐标。Vector3对象,面与面有共用的顶点,所以为了节约内存,先存顶点,然后再存三角形;

关于Unity中Mesh网格的详解
5: 面---->三角形索引数组<int>: Mesh里面每个三角形为一个面,由于面与面的顶点公用,所以,用索引来表示三角形的一个面,可以节约模型内存空间, 0, 1, 2表示一个面,对应的顶点是在顶点数组中的索引,三角形顶点的顺序为逆时针为正面,顺时针为反面。

关于Unity中Mesh网格的详解
6: 顶点法线: 面的法线是与面垂直的线, 严格意义上讲,点是没有法线的, 在光照计算的时候,使用法线来进行光照计算,

 如果一个面上所有的法线都是一样,那么光着色也一样,看起来会很奇怪,所以通过某种算法,把多个面公用的顶点的法线根据算法综合插值,得到顶点法线;
7: 顶点纹理坐标<Vector2>: 顶点对应的纹理上的UV坐标;
6: 顶点切线<Vector4> 顶点切线,知道有这个东西就行了;

 

 

Mesh的重要属性

(1) vertices 网格顶点数组;
(2) normals 网格的法线数组;
(3) tangents 网格的切线数组;
(4) uv 网格的基础纹理坐标;
(5) uv2 网格设定的第二个纹理坐标;
(6) bounds 网格的包围盒;
(7) Colors 网格的顶点颜色数组;
(8) triangles 包含所有三角形的顶点索引数组;
(9) vectexCount 网格中的顶点数量(只读的);
(10) subMeshCount 子网格的数量,每个材质都有一个独立的网格列表;
(11) bonesWeights: 每个顶点的骨骼权重;
(12) bindposes: 绑定姿势,每个索引绑定的姿势使用具有相同的索引骨骼;

 

 

Mesh的重要方法

(1) Clear 清空所有的顶点数据和所有的三角形索引;
(2) RecalculateBounds 重新计算网格的包围盒;
(3) RecalculateNormals 重新计算网格的法线;
(4) Optimze 显示优化的网格;
(5) GetTriangles 返回网格的三角形列表;
(6) SetTriangles 为网格设定三角形列表;
(7) CominMeshes组合多个网格到同一个网格;

 

 

Mesh修改案例

1: 将模型的Mesh复制给Mesh filter组件的Mesh数据。
2: 讲模型的Mesh的模型顶点数和面数增加;
3: 开发思路:
  (1) 创建项目,配置目录,导入模型,材质;
  (2) 模型拖入场景树,去掉其他的组件,只保留Mesh filter,点击里面的实例查看Mesh;
  (3) 创建一个空的节点,加入Mesh filter组件,加入MeshRender组件,关联好材质;
  (4) 创建脚本,挂载到这个空节点上,脚本上有组件Mesh filter,关联到前面有的Mesh节点;
  (5) 赋值顶点,三角形, 法线,切线,纹理坐标, 运行观察结果;
  (6) 插值顶点,法线,切线, 纹理坐标, 重新设置三角形索引, 运行观察结果;

 

 

Mesh案例详细步骤

1.创建Unity工程和文件目录

2.导入模型和材质到res文件夹下zhang.FBX和wenli.tga(第54)

3.把模型拖入场景中,点击模型的Mesh Filter组件的Mesh属性,发现多一个资源出来,那个就是过滤读取到的网格,可以查看详细的网格属性

24个顶点,12个三角形

关于Unity中Mesh网格的详解

4.模型的Mesh Renderer组件是用来绘制网格的组件,它的Mesh是Mesh Filter传递过来的,如果隐藏这个组件,场景中就不会显示出模型

5.创建一个材质,把wenli.tga当做材质的纹理贴图拖进Albedo里面,然后把模型和材质关联。

6.效果

关于Unity中Mesh网格的详解

 

 

代码获得Mesh

1.创建一个空节点item,添加一个Mesh Filter组件

2.创建一个脚本mesh_test,挂载在item下面,通过代码来获得其他模型的Mesh

打开mesh_test

using UnityEngine;
using System.Collections;public class mesh_test : MonoBehaviour {
    public MeshFilter cube_mesh;//获得编辑器传递进来的模型的MeshFilter组件,必须是已经有MeshFilter组件和Mesh的节点
    // Use this for initialization
    void Start () {
        Mesh cube = this.cube_mesh.mesh;//传递进来的模型的MeshFilter组件的Mesh赋值给Mesh类型的变量cube
        
        Mesh self_mesh = this.GetComponent<MeshFilter>().mesh;//获得自己节点下的MeshFilter组件过滤得到的Mesh
        self_mesh.Clear();//先把自己的Mesh清零
        self_mesh.vertices = cube.vertices;//把变量cube的顶点数组传递给自己
        self_mesh.triangles = cube.triangles;//把变量cube的三角形数组传递给自己
        self_mesh.normals = cube.normals;//把变量cube的法线数组传递给自己
        self_mesh.uv = cube.uv;//把变量cube的纹理坐标数组传递给自己
        self_mesh.tangents = cube.tangents;//把变量cube的切线数组传递给自己

        self_mesh.RecalculateBounds();//重新计算自己的Mesh
       }

    // Update is called once per frame
    void Update () {
    
    }
}

3.再给item添加Mesh Renderer组件,再关联一个材质,这样,它就可以在场景中绘制出模型了,它的Mesh是别人那里拿的。

 

 

复杂操作Mesh

1.思路

把模型中的所有三角形都再增加三个顶点,每个顶点在对应边的中点。

关于Unity中Mesh网格的详解

2.创建一个脚本mesh_test,挂载在item下面,通过代码来增加顶点

打开脚本mesh_test

using UnityEngine;
using System.Collections;
using System.Collections.Generic;

public class mesh_test : MonoBehaviour {
    public MeshFilter cube_mesh;
    // Use this for initialization
    void Start () {
        Mesh cube = this.cube_mesh.mesh;

        //定义需要用到的和Mesh有关的变量
        List<Vector3> vertices = new List<Vector3>();
        List<int> triangles = new List<int>();
        List<Vector3> normals = new List<Vector3>();
        List<Vector2> uv = new List<Vector2>();
        List<Vector4> tangents = new List<Vector4>();


        //遍历Mesh的三角形数组
        for (int i = 0; i < cube.triangles.Length / 3; i++) {//一个模型包含非常多的三角形,每个三角形都要执行我们定义的复杂操作
            Vector3 t0 = cube.vertices[cube.triangles[i * 3 + 0]];//得到第一个顶点的坐标
            Vector3 t1 = cube.vertices[cube.triangles[i * 3 + 1]];//得到第二个顶点的坐标
            Vector3 t2 = cube.vertices[cube.triangles[i * 3 + 2]];//得到第三个顶点的坐标

            Vector3 t3 = Vector3.Lerp(t0, t1, 0.5f);//第三个点的坐标为第一个点和第二个点的中点
            Vector3 t4 = Vector3.Lerp(t1, t2, 0.5f);//第四个点的坐标为第二个点和第三个点的中点
            Vector3 t5 = Vector3.Lerp(t0, t2, 0.5f);//第五个点的坐标为第一个点和第三个点的中点

            int count = vertices.Count;//获得初始的大小,等下用这个变量可以表示索引

            //插入顶点坐标到顶点数组vertices中,vertices填充完毕
            vertices.Add(t0); // 索引为count + 0
            vertices.Add(t1); // 索引为count + 1
            vertices.Add(t2); // 索引为count + 2
            vertices.Add(t3); // 索引为count + 3
            vertices.Add(t4); // 索引为count + 4
            vertices.Add(t5); // 索引为count + 5


            //-------------------------------------------------------------
            //插入三角形顶点索引到三角形数组triangles中,triangles填充完毕
            triangles.Add(count + 0); triangles.Add(count + 3); triangles.Add(count + 5);
            triangles.Add(count + 3); triangles.Add(count + 1); triangles.Add(count + 4);
            triangles.Add(count + 4); triangles.Add(count + 2); triangles.Add(count + 5);
            triangles.Add(count + 3); triangles.Add(count + 4); triangles.Add(count + 5);

            //-------------------------------------------------------------
            //和上面获得顶点坐标的做法一样,获得各个normals法线坐标
            Vector3 n0 = cube.normals[cube.triangles[i * 3 + 0]];
            Vector3 n1 = cube.normals[cube.triangles[i * 3 + 1]];
            Vector3 n2 = cube.normals[cube.triangles[i * 3 + 2]];

            Vector3 n3 = Vector3.Lerp(n0, n1, 0.5f);
            Vector3 n4 = Vector3.Lerp(n1, n2, 0.5f);
            Vector3 n5 = Vector3.Lerp(n0, n2, 0.5f);

            //插入法线坐标到法线数组normals中,normals填充完毕
            normals.Add(n0); 
            normals.Add(n1); 
            normals.Add(n2); 
            normals.Add(n3); 
            normals.Add(n4); 
            normals.Add(n5);

            //-------------------------------------------------------------
            //和上面获得顶点坐标的做法一样,获得各个uv纹理坐标
            Vector2 uv0 = cube.uv[cube.triangles[i * 3 + 0]];
            Vector2 uv1 = cube.uv[cube.triangles[i * 3 + 1]];
            Vector2 uv2 = cube.uv[cube.triangles[i * 3 + 2]];

            Vector2 uv3 = Vector3.Lerp(uv0, uv1, 0.5f);
            Vector2 uv4 = Vector3.Lerp(uv1, uv2, 0.5f);
            Vector2 uv5 = Vector3.Lerp(uv0, uv2, 0.5f);

            //插入纹理坐标到纹理数组uv中,uv填充完毕
            uv.Add(uv0);
            uv.Add(uv1);
            uv.Add(uv2);
            uv.Add(uv3);
            uv.Add(uv4);
            uv.Add(uv5);

            //-------------------------------------------------------------
            //和上面获得顶点坐标的做法一样,获得各个tangents切线坐标
            Vector4 tan0 = cube.tangents[cube.triangles[i * 3 + 0]];
            Vector4 tan1 = cube.tangents[cube.triangles[i * 3 + 1]];
            Vector4 tan2 = cube.tangents[cube.triangles[i * 3 + 2]];

            Vector4 tan3 = Vector3.Lerp(tan0, tan1, 0.5f);
            Vector4 tan4 = Vector3.Lerp(tan1, tan2, 0.5f);
            Vector4 tan5 = Vector3.Lerp(tan0, tan2, 0.5f);

            //插入切线坐标到切线数组tangents中,tangents填充完毕
            tangents.Add(tan0);
            tangents.Add(tan1);
            tangents.Add(tan2);
            tangents.Add(tan3);
            tangents.Add(tan4);
            tangents.Add(tan5);
        }

        //传递给自己的Mesh并重新绘制网格
        Mesh self_mesh = this.GetComponent<MeshFilter>().mesh;
        self_mesh.Clear();
        self_mesh.vertices = vertices.ToArray();//List转换为Array
        self_mesh.triangles = triangles.ToArray();
        self_mesh.normals = normals.ToArray();
        self_mesh.uv = uv.ToArray();
        self_mesh.tangents = tangents.ToArray();

        self_mesh.RecalculateBounds();

        //没有删除重复的顶点,有待完善
    }
    
    // Update is called once per frame
    void Update () {
    
    }
}

3.效果

关于Unity中Mesh网格的详解      关于Unity中Mesh网格的详解

相关资源:Unity 网格合并Mesh Combine 1.61

上一篇:OVN学习整理


下一篇:OVN实战一之GNS3操作指南及OVN入门