如何为基于图块的几何图形使用更复杂的网格简化运行时网格?

时间:2019-02-01 19:31:51

标签: c# unity3d mesh

我正在使用此网格简化器:

using System;
using System.Collections.Generic;
using System.Text;

using UnityEngine;
using UnityEditor;

static public class MeshColliderTools {

    public static void SnapToGrid(this Mesh mesh, float gridDelta) {
        if (gridDelta < 1e-5f)
            return;
        float inverse = 1f / gridDelta;
        var verts = mesh.vertices;
        for (int i = 0; i < verts.Length; i++) {
            verts[i].x = Mathf.RoundToInt(verts[i].x*inverse) / inverse;
            verts[i].y = Mathf.RoundToInt(verts[i].y*inverse) / inverse;
            verts[i].z = Mathf.RoundToInt(verts[i].z*inverse) / inverse;
        }
        mesh.vertices = verts;
    }

    public static void Weld (this Mesh mesh, float threshold, float bucketStep) {
        Vector3[] oldVertices = mesh.vertices;
        Vector3[] newVertices = new Vector3[oldVertices.Length];
        int[] old2new = new int[oldVertices.Length];
        int newSize = 0;

        // Find AABB
        Vector3 min = new Vector3 (float.MaxValue, float.MaxValue, float.MaxValue);
        Vector3 max = new Vector3 (float.MinValue, float.MinValue, float.MinValue);
        for (int i = 0; i < oldVertices.Length; i++) {
            if (oldVertices[i].x < min.x) min.x = oldVertices[i].x;
            if (oldVertices[i].y < min.y) min.y = oldVertices[i].y;
            if (oldVertices[i].z < min.z) min.z = oldVertices[i].z;
            if (oldVertices[i].x > max.x) max.x = oldVertices[i].x;
            if (oldVertices[i].y > max.y) max.y = oldVertices[i].y;
            if (oldVertices[i].z > max.z) max.z = oldVertices[i].z;
        }
        min -= Vector3.one * 0.111111f;
        max += Vector3.one * 0.899999f;

        // Make cubic buckets, each with dimensions "bucketStep"
        int bucketSizeX = Math.Max(1, Mathf.FloorToInt ((max.x - min.x) / bucketStep) + 1);
        int bucketSizeY = Math.Max(1, Mathf.FloorToInt ((max.y - min.y) / bucketStep) + 1);
        int bucketSizeZ = Math.Max(1, Mathf.FloorToInt ((max.z - min.z) / bucketStep) + 1);
        List<int>[,,] buckets = new List<int>[bucketSizeX, bucketSizeY, bucketSizeZ];

        // Make new vertices
        for (int i = 0; i < oldVertices.Length; i++) {
            // Determine which bucket it belongs to
            int x = Mathf.FloorToInt ((oldVertices[i].x - min.x) / bucketStep);
            int y = Mathf.FloorToInt ((oldVertices[i].y - min.y) / bucketStep);
            int z = Mathf.FloorToInt ((oldVertices[i].z - min.z) / bucketStep);

            // Check to see if it's already been added
            if (buckets[x, y, z] == null)
                buckets[x, y, z] = new List<int> (); // Make buckets lazily

            for (int j = 0; j < buckets[x, y, z].Count; j++) {
                Vector3 to = newVertices[buckets[x, y, z][j]] - oldVertices[i];
                if (Vector3.SqrMagnitude (to) < 0.001f) {
                    old2new[i] = buckets[x, y, z][j];
                    goto skip; // Skip to next old vertex if this one is already there
                }
            }

            // Add new vertex
            newVertices[newSize] = oldVertices[i];
            buckets[x, y, z].Add (newSize);
            old2new[i] = newSize;
            newSize++;

            skip:;
        }

        // Make new triangles
        int[] oldTris = mesh.triangles;
        int[] newTris = new int[oldTris.Length];
        for (int i = 0; i < oldTris.Length; i++) {
            newTris[i] = old2new[oldTris[i]];
        }

        Vector3[] finalVertices = new Vector3[newSize];
        for (int i = 0; i < newSize; i++)
            finalVertices[i] = newVertices[i];

        mesh.Clear();
        mesh.vertices = finalVertices;
        mesh.triangles = newTris;

        // Debug.LogFormat("Weld vert count: {0} vs. {1}", newSize, oldVertices.Length);
    }


    public static void Simplify(this MeshCollider meshCollider) { meshCollider.sharedMesh.Simplify(); }
    public static void Simplify(this Mesh mesh) {
        var verts = mesh.vertices;
        var origNumVerts = verts.Length;

        var workingSet = new List<Vertice>(origNumVerts);
        for (int i = 0; i < origNumVerts; i++) {
            var r = new Vertice();
            r.position = verts[i];
            workingSet.Add(r);
        }

        var tris = mesh.triangles;
        var triLength = tris.Length;
        for (int i = 0; i < triLength; i+=3)
            Face.AddFace(workingSet, tris, i);

        for (int i = 0; i < origNumVerts; i++)
            workingSet[i].AssignLinearPosition();


        /*********************************
         *                               *
         *         Simplify mesh!        *
         *                               *
         ********************************/

        HashSet<Vertice> candidates;
        HashSet<Vertice> nextCandidates  = new HashSet<Vertice>();

        foreach (Vertice v in workingSet) if (!v.IsStatic)
            nextCandidates.Add(v);

        while (nextCandidates.Count != 0) {
            candidates = nextCandidates;
            nextCandidates = new HashSet<Vertice>();
            foreach (Vertice a in candidates) {
                if (a.edges != null) foreach (Edge ac in a.edges) {
                        Vertice c;
                        if (a.CanFollow(ac, out c)) {
                            foreach (Face f in ac.faces)
                                Edge.Collapse(f.GetOpposite(c), f.GetOpposite(a), f);
                            for (int i = a.edges.Count-1; i >= 0; --i) {
                            //foreach (Edge edge_of_a in a.edges) {
                                if (a.edges[i] != ac) {
                                    var o = a.edges[i].GetOpposite(a);
                                    if (!o.IsStatic) nextCandidates.Add(o);
                                    a.edges[i].Reconnect(a, c);
                                }
                            }
                            if (!c.IsStatic) nextCandidates.Add(c);

                            c.DisconnectFrom(ac);
                            a.Disconnect();
                            ac.DisconnectIncludingFaces();
                            break;
                        }
                    }
            }
        }

        var simplifiedVerts = new List<Vertice>();
        foreach (Vertice v in workingSet) if (v.edges != null)
            simplifiedVerts.Add(v);

        var simplifiedNumVerts = simplifiedVerts.Count;
        var newPositions = new Vector3[simplifiedNumVerts];
        //var resultColors = new Color[simplifiedNumVerts];
        for (int i = 0; i < simplifiedNumVerts; i++) {
            simplifiedVerts[i].finalIndex = i;
            newPositions[i] = simplifiedVerts[i].position;
        }

        var resultTris = new List<int>();
        var triSet = new HashSet<Face>();
        foreach (Vertice v in simplifiedVerts) {
            foreach (Edge e in v.edges) {
                foreach (Face f in e.faces) if (!triSet.Contains(f)) {
                        triSet.Add(f);
                        f.GetIndexes(resultTris);
                    }
            }
        }

        mesh.Clear();
        mesh.vertices = newPositions;
        mesh.triangles = resultTris.ToArray();

        mesh.RecalculateBounds();
        ;

        // Debug.LogFormat("Simplify vert count: {0} vs. {1}", simplifiedNumVerts, origNumVerts);
    }

    private class Vertice {
        public List<Edge> edges = new List<Edge>();
        public Vector3 position;

        /// <summary>A cache used for identifying a vertice during mesh reconstruction.
        public int finalIndex;

        /// <summary>
        ///  - Null if vertice is internal in a plane
        ///  - A non-zero vector if vertice is internal in a line
        ///  - Vector3.zero otherwise.
        /// </summary>
        public Vector3? linearPosition;

        public void AssignLinearPosition() {
            for (int i = 0; i < edges.Count; i++) {
                var edge = edges[i];
                if (!edge.HasEqualPlanes()) {
                    if (linearPosition == null)
                        linearPosition = edge.vertices[1].position - edge.vertices[0].position;
                    else if (!edge.IsParallel(linearPosition)) {
                        linearPosition = Vector3.zero;
                        break;
                    }
                }
            }
        }

        public bool IsStatic { get {
                return linearPosition == Vector3.zero;
            } }

        public Edge GetExistingConnectingEdge(Vertice v) {
            foreach (Edge e in edges) {
                if (e.vertices[0] == this && e.vertices[1] == v) return e;
                if (e.vertices[1] == this && e.vertices[0] == v) return e;
            }
            return null;
        }

        public Edge GetConnectingEdge(Vertice v) {
            Edge result = GetExistingConnectingEdge(v);
            if (result == null) {
                result = new Edge(this, v);
                edges.Add(result);
                v.edges.Add(result);
            }
            return result;
        }

        /// <summary>When collapsing an edge a->c, the linear space must
        /// be respected, and all faces, not connected with a->c,
        /// must not flip.
        /// </summary>
        public bool CanFollow(Edge transportEdge, out Vertice opposite) {
            if (IsStatic) { opposite = default(Vertice); return false; }
            if (linearPosition != null && !transportEdge.IsParallel(linearPosition)) { opposite = default(Vertice); return false; }

            var localTris = new HashSet<Face>();
            foreach (Edge e in edges) foreach (Face f in e.faces)
                localTris.Add(f);

            localTris.ExceptWith(transportEdge.faces);

            opposite = transportEdge.GetOpposite(this);
            var targetPos = opposite.position;
            var lTriEnum = localTris.GetEnumerator();
            try {
                while (lTriEnum.MoveNext())
                    if (lTriEnum.Current.MoveWouldFlip(this, targetPos))
                        return false;
            } finally { lTriEnum.Dispose(); }
            return true;
        }

        public void DisconnectFrom(Edge e) {
            edges.Remove(e);
        }

        public void Disconnect() {
            edges.Clear();
            edges = null;
        }

    }

    private class Edge {
        public List<Vertice> vertices;
        public List<Face> faces;

        public static void Collapse(Edge moved, Edge target, Face f) {
            Face faceOutsideMoved;
            try { faceOutsideMoved = moved.GetOpposite(f); }
            catch (Exception e) {
                foreach (Vertice v in moved.vertices) {
                    v.edges.Remove(moved);
                }
                //throw new Exception(e.Message + "\n" + moved.vertices[0].position + " <--> " + moved.vertices[1].position);
                return;
            }
            faceOutsideMoved.Replace(moved, target);
            target.Replace(f, faceOutsideMoved);
            foreach (Vertice v in moved.vertices) {
                v.edges.Remove(moved);
            }
        }

        public Edge(Vertice v0, Vertice v1) {
            vertices = new List<Vertice>(2);
            vertices.Add(v0);
            vertices.Add(v1);
            faces = new List<Face>(2);
        }

        public Vertice GetOpposite(Vertice v) {
            var v0 = vertices[0];
            return v != v0 ? v0 : vertices[1];
        }

        public Face GetOpposite(Face v) {
            if (faces.Count != 2) {
                throw new Exception ("Collapsing an edge with only 1 face into another. This is not supported.");
            }
            var face0 = faces[0];
            return face0 == v ? faces[1] : face0;
        }

        public bool HasEqualPlanes() {
            if (faces.Count != 2) return false;
            var f0   = faces[0];
            var f0e0 = faces[0].edges[0];

            var f1   = faces[1];
            var f1e0 = faces[1].edges[0];

            var e0 = f0e0 != this ? f0e0 : f0.edges[1];
            var e1 = f1e0 != this ? f1e0 : f1.edges[1];

            var v0 =    vertices[1].position -    vertices[0].position;
            var v1 = e0.vertices[1].position - e0.vertices[0].position;
            var v2 = e1.vertices[1].position - e1.vertices[0].position;

            var n0 = Vector3.Cross(v0, v1);

            var dot = Vector3.Dot(n0, v2);
            return -5e-3 < dot && dot < 5e-3;
        }

        public void Replace (Face oldFace, Face newFace) {
            for (int j = 0; j < faces.Count; j++) {
                if (faces[j] == oldFace) {
                    faces[j] = newFace;
                    return;
                }
            }
        }

        public void Reconnect(Vertice oldVertice, Vertice newVertice) {
            if (vertices[0] == oldVertice) vertices[0] = newVertice;
            else                           vertices[1] = newVertice;
            newVertice.edges.Add(this);
        }

        public bool Contains(Vertice v) {
            return v == vertices[0] || v == vertices[1];
        }

        public bool IsParallel(Vector3? nv) {
            var v0 = vertices[0].position;
            var v1 = vertices[1].position;
            float cross = Vector3.Cross(v1 - v0, nv.Value).sqrMagnitude;
            return -5e-6f < cross && cross < 5e-6f;
        }

        public void DisconnectIncludingFaces() {
            vertices.Clear();
            vertices = null;
            foreach (Face f in faces)
                f.Disconnect();
            faces.Clear();
            faces = null;
        }
    }

    private class Face {
        public Edge[] edges;

        /*
         * <paramref name="e0">Edge between v0 and v1</paramref>
         * <paramref name="e1">Edge between v0 and v2</paramref>
         * <paramref name="e2">Edge between v1 and v2</paramref>
         */
        private Face(Edge e0, Edge e1, Edge e2) {
            edges = new Edge[] { e0, e1, e2 };
        }

        /*
         * <paramref name="allVertices">List of vertices, in the same order as 'verts' from the mesh.</paramref>
         * <paramref name="tris">The tris array from the mesh.</paramref>
         * <paramref name="triIndex">The index of the first vertex in this triangle.</paramref>
         */
        public static void AddFace(List<Vertice> allVertices, int[] tris, int triIndex) {
            var v0 = allVertices[tris[triIndex+0]];
            var v1 = allVertices[tris[triIndex+1]];
            var v2 = allVertices[tris[triIndex+2]];

            var e0 = v0.GetConnectingEdge(v1);
            var e1 = v0.GetConnectingEdge(v2);
            var e2 = v1.GetConnectingEdge(v2);

            var face = new Face(e0, e1, e2);

            e0.faces.Add(face);
            e1.faces.Add(face);
            e2.faces.Add(face);
        }

        public Edge GetOpposite(Vertice v) {
            Edge e0, e1;
            if (!(e0 = edges[0]).Contains(v)) return e0;
            if (!(e1 = edges[1]).Contains(v)) return e1;
            return edges[2];
        }

        public Vertice GetOpposite(Edge o) {
            var o0 = o.vertices[0];
            var o1 = o.vertices[1];
            for (int i = 0; i < 3; i++) { // Will never reach 3 because there will be an unequal vertice before the third edge
                Edge e = edges[i];
                Vertice e0 = e.vertices[0];
                if (e0 != o0 && e0 != o1) return e0;
                Vertice e1 = e.vertices[1];
                if (e1 != o0 && e1 != o1) return e1;
            }
            throw new Exception("A face seems to have three edges that all share a vertice with a given edge.");
        }

        public void Replace (Edge oldEdge, Edge newEdge) {
            if      (edges[0] == oldEdge) edges[0] = newEdge;
            else if (edges[1] == oldEdge) edges[1] = newEdge;
            else                          edges[2] = newEdge;
        }

        public bool MoveWouldFlip(Vertice v, Vector3 p) {
            Edge oppositeEdge = GetOpposite(v);
            var ov0 = oppositeEdge.vertices[0].position;
            var ov  = oppositeEdge.vertices[1].position - ov0;
            var ot = p          - ov0;
            var ct = v.position - ov0;

            var cross0 = Vector3.Cross(ot, ov);
            var c0SqrMagnitude = cross0.sqrMagnitude;
            if (c0SqrMagnitude < 0.0001f) return true;

            var cross1 = Vector3.Cross(ct, ov);
            var c1SqrMagnitude = cross1.sqrMagnitude;
            if (c1SqrMagnitude < 0.0001f) return true;

            if (Mathf.Sign(Vector3.Dot(cross0, cross1)) < 0f) return true;

            return false;
        }

        /*
         * <summary>Adds the finalIndex for the verts in order: v0, v1, v2.</summary>
         */
        public void GetIndexes(List<int> results) {
            results.Add(GetOpposite(edges[2]).finalIndex);
            results.Add(GetOpposite(edges[1]).finalIndex);
            results.Add(GetOpposite(edges[0]).finalIndex);
        }

        public void Disconnect() {
            edges = null;
        }
    }
}

如果网格很简单并且使用多维数据集,则可以很好地简化基于图块的网格,如下所示:

Mesh Image

但是,如果我尝试将其用于更复杂的网格,则会出错。

  

例外:一张脸似乎有三个边缘,这些边缘都与给定的边缘共享一个顶点

以及一些MeshColliderTools.cs:350 -> MeshColliderTools.cs:143MeshColliderTools.cs:293 -> MeshColliderTools.cs:137MeshColliderTools.cs:415 -> MeshColliderTools.cs:440 -> MeshColliderTools.cs:265 -> MeshColliderTools.cs:135的空引用

这是更复杂的网格的样子:

Complicated Mesh

1 个答案:

答案 0 :(得分:0)

网格抽取/简化是最难解决的问题之一。有太多可能会破坏代码的边缘情况。

在您的代码中,您似乎已经在考虑边界边缘的可能性(仅附着了一个小平面的边缘),但是您并未考虑非流形边缘的可能性(边缘更多的边缘)超过2个构面)。我看到您有几个断言,表示面数== 2,从理论上讲可以防止非流形边缘的问题。

但是,可以通过折叠边缘来产生非歧管边缘。如果要折叠的局部区域与磁盘不是同胚的,则总是会发生这种情况(即,您不能将局部几何图形平放在折叠边缘周围)。

每当我编写网格处理代码时,我的调试策略就始终相同:

保存每次折叠的迭代,然后查看成功折叠的最后一个文件。您很可能会看到非流形几何。追溯到非流形边缘插入的位置,然后在调试器中逐行运行该折叠,并找到代码允许通过的位置。然后,修补代码的这一部分以解决非歧管边缘问题,然后等到找到另一个完全破坏代码的网格。

根据我的经验,我发现最好只使用别人现有的网格结构。 This StackOverflow link提到了声称是健壮的开源实现。可能值得一看。