Version: 2023.2
使用 Mesh 类
网格的细节级别 (LOD)

Example: creating a quad

为了表示平面,Unity 包含了可以在场景中实例化的平面和四边形原始游戏对象。但是,了解如何使用脚本来自行构造四边形网格很有用。这是进行程序化网格生成的基础。

注意:Unity 会处理并显示由三角形组成的几何体,而不是由四边形组成的几何体。这意味着一个四边形图元包含两个三角形。

顶点数组

首先需要设置形状使用的顶点数组。

以下示例假定四边形位于 x 轴和 y 轴上,并且脚本中包含变量 width 和 height。

Vector3[] vertices = new Vector3[4]
{
    new Vector3(0, 0, 0),
    new Vector3(width, 0, 0),
    new Vector3(0, height, 0),
    new Vector3(width, height, 0)
};
mesh.vertices = vertices;

此示例按以下顺序提供顶点: 左下角、 右下角、 左上角、 右上角

由于 Unity 检索网格数据属性的方式,您在自己的数组中设置数据并随后将该数组分配给属性(例如,分配给 Mesh.verticesMesh.normals)的效率要高得多,而不要通过单个元素访问属性数组。

三角形

接下来需要设置三角形。一个四边形由两个三角形组成,每个三角形由先前创建的顶点数组中的三个点组成。要指定这些点,请将每个三角形定义为顶点数组的三个索引。例如,此四边形的左下方三角形使用索引 0、2 和 1,对应于顶点数组中的坐标 (0, 0, 0)、(0, height, 0) 和 (width, 0, 0)。顺序很重要,因为必须按顺时针对角进行排序。右上方的三角形使用索引 2、3 和 1。

int[] tris = new int[6]
{
    // 左下方三角形
    0, 2, 1,
    // 右上方三角形
    2, 3, 1
};
mesh.triangles = tris;

法线

在场景中可以看到带有顶点和三角形的网格,但是 Unity 尚未对网格正确着色,因为这个网格还没有法线。此示例的法线很简单,因为这些法线都是相同的。每条法线均指向四边形本地空间中的负 z 轴方向。添加法线时,Unity 会正确对四边形着色,但是您需要在场景中有光源才能看到效果。

Vector3[] normals = new Vector3[4]
{
    -Vector3.forward,
    -Vector3.forward,
    -Vector3.forward,
    -Vector3.forward
};
mesh.normals = normals;

If you do not want to define the normals yourself, you can use Mesh.RecalculateNormals().

纹理坐标

最后,要正确显示网格的材质上的纹理,请向网格添加纹理坐标。纹理坐标在 0 到 1 之间。网格中的每个顶点都有一个纹理坐标,用于指定材质的纹理上要采样的位置。要显示四边形上的整个纹理,每个顶点上的纹理坐标值都应全部为 0 或 1,以便四边形的每个角都对应于纹理的角。

Vector2[] uv = new Vector2[4]
{
      new Vector2(0, 0),
      new Vector2(1, 0),
      new Vector2(0, 1),
      new Vector2(1, 1)
};
mesh.uv = uv;

最终的脚本

以下脚本结合了以上所有内容,用于在场景中创建一个四边形。要使用该脚本: 创建一个新的 C# 脚本(菜单:Assets > Create > C# Script),然后将其命名为 QuadCreator。 打开 QuadCreator 脚本,将示例代码复制到其中,然后保存脚本。 回到 Editor 中,在场景中创建一个新的游戏对象(菜单:GameObject > Create Empty)。 在 Inspector 中,选择 Add Component > Scripts > Quad Creator。 将游戏对象放置在场景中所需的任何位置。 进入运行模式。如果在 Scene 视图或 Game 视图中看不到该四边形,请确保从正确的一侧查看;Unity 不会渲染此网格的背面。

using UnityEngine;

public class QuadCreator : MonoBehaviour
{
    public float width = 1;
    public float height = 1;

    public void Start()
    {
        MeshRenderer meshRenderer = gameObject.AddComponent<MeshRenderer>();
        meshRenderer.sharedMaterial = new Material(Shader.Find("Standard"));

        MeshFilter meshFilter = gameObject.AddComponent<MeshFilter>();

        Mesh mesh = new Mesh();

        Vector3[] vertices = new Vector3[4]
        {
            new Vector3(0, 0, 0),
            new Vector3(width, 0, 0),
            new Vector3(0, height, 0),
            new Vector3(width, height, 0)
        };
        mesh.vertices = vertices;

        int[] tris = new int[6]
        {
            // 左下方三角形
            0, 2, 1,
            // 右上方三角形
            2, 3, 1
        };
        mesh.triangles = tris;

        Vector3[] normals = new Vector3[4]
        {
            -Vector3.forward,
            -Vector3.forward,
            -Vector3.forward,
            -Vector3.forward
        };
        mesh.normals = normals;

        Vector2[] uv = new Vector2[4]
        {
            new Vector2(0, 0),
            new Vector2(1, 0),
            new Vector2(0, 1),
            new Vector2(1, 1)
        };
        mesh.uv = uv;

        meshFilter.mesh = mesh;
    }
}

注意:此示例代码在 Start 函数中,这意味着会在您进入运行模式时执行一次,并且网格在整个应用程序中不会改变。但是,您可以在 Update 函数中添加代码,以使网格在每一帧都更改。请注意,这会大大增加生成网格时消耗资源的强度。

使用 Mesh 类
网格的细节级别 (LOD)