using System; using UnityEngine; using System.Linq; using System.Collections.Generic; using UnityEngine.ProBuilder; namespace UnityEngine.ProBuilder.MeshOperations { /// /// Face and edge extrusion. /// public static class ExtrudeElements { /// /// Extrude a collection of faces. /// /// The source mesh. /// The faces to extrude. /// Describes how faces are extruded. /// The distance to extrude faces. /// An array of the faces created as a result of the extrusion. Null if the faces paramater is null or empty. public static Face[] Extrude(this ProBuilderMesh mesh, IEnumerable faces, ExtrudeMethod method, float distance) { switch (method) { case ExtrudeMethod.IndividualFaces: return ExtrudePerFace(mesh, faces, distance); default: return ExtrudeAsGroups(mesh, faces, method == ExtrudeMethod.FaceNormal, distance); } } /// /// Extrude a collection of edges. /// /// The source mesh. /// The edges to extrude. /// The distance to extrude. /// If true adjacent edges will be extruded retaining a shared vertex, if false the shared vertex will be split. /// Pass true to allow this function to extrude manifold edges, false to disallow. /// The extruded edges, or null if the action failed due to manifold check or an empty edges parameter. public static Edge[] Extrude(this ProBuilderMesh mesh, IEnumerable edges, float distance, bool extrudeAsGroup, bool enableManifoldExtrude) { if (mesh == null) throw new ArgumentNullException("mesh"); if (edges == null) throw new ArgumentNullException("edges"); SharedVertex[] sharedIndexes = mesh.sharedVerticesInternal; List validEdges = new List(); List edgeFaces = new List(); foreach (Edge e in edges) { int faceCount = 0; Face fa = null; foreach (Face face in mesh.facesInternal) { if (mesh.IndexOf(face.edgesInternal, e) > -1) { fa = face; if (++faceCount > 1) break; } } if (enableManifoldExtrude || faceCount < 2) { validEdges.Add(e); edgeFaces.Add(fa); } } if (validEdges.Count < 1) return null; Vector3[] localVerts = mesh.positionsInternal; if (!mesh.HasArrays(MeshArrays.Normal)) mesh.Refresh(RefreshMask.Normals); IList oNormals = mesh.normals; int[] allEdgeIndexes = new int[validEdges.Count * 2]; int c = 0; for (int i = 0; i < validEdges.Count; i++) { allEdgeIndexes[c++] = validEdges[i].a; allEdgeIndexes[c++] = validEdges[i].b; } List extrudedIndexes = new List(); // used to set the editor selection to the newly created edges List newEdges = new List(); bool hasColors = mesh.HasArrays(MeshArrays.Color); // build out new faces around validEdges for (int i = 0; i < validEdges.Count; i++) { Edge edge = validEdges[i]; Face face = edgeFaces[i]; // Averages the normals using only vertices that are on the edge Vector3 xnorm = extrudeAsGroup ? InternalMeshUtility.AverageNormalWithIndexes(sharedIndexes[mesh.GetSharedVertexHandle(edge.a)], allEdgeIndexes, oNormals) : Math.Normal(mesh, face); Vector3 ynorm = extrudeAsGroup ? InternalMeshUtility.AverageNormalWithIndexes(sharedIndexes[mesh.GetSharedVertexHandle(edge.b)], allEdgeIndexes, oNormals) : Math.Normal(mesh, face); int x_sharedIndex = mesh.GetSharedVertexHandle(edge.a); int y_sharedIndex = mesh.GetSharedVertexHandle(edge.b); var positions = new Vector3[4] { localVerts[edge.a], localVerts[edge.b], localVerts[edge.a] + xnorm.normalized * distance, localVerts[edge.b] + ynorm.normalized * distance }; var colors = hasColors ? new Color[4] { mesh.colorsInternal[edge.a], mesh.colorsInternal[edge.b], mesh.colorsInternal[edge.a], mesh.colorsInternal[edge.b] } : null; Face newFace = mesh.AppendFace( positions, colors, new Vector2[4], new Vector4[4], new Vector4[4], new Face(new int[6] { 2, 1, 0, 2, 3, 1 }, face.submeshIndex, AutoUnwrapSettings.tile, 0, -1, -1, false), new int[4] { x_sharedIndex, y_sharedIndex, -1, -1 }); newEdges.Add(new Edge(newFace.indexesInternal[3], newFace.indexesInternal[4])); extrudedIndexes.Add(new Edge(x_sharedIndex, newFace.indexesInternal[3])); extrudedIndexes.Add(new Edge(y_sharedIndex, newFace.indexesInternal[4])); } // merge extruded vertex indexes with each other if (extrudeAsGroup) { for (int i = 0; i < extrudedIndexes.Count; i++) { int val = extrudedIndexes[i].a; for (int n = 0; n < extrudedIndexes.Count; n++) { if (n == i) continue; if (extrudedIndexes[n].a == val) { mesh.SetVerticesCoincident(new int[] { extrudedIndexes[n].b, extrudedIndexes[i].b }); break; } } } } // todo Should only need to invalidate caches on affected faces foreach (Face f in mesh.facesInternal) f.InvalidateCache(); return newEdges.ToArray(); } /// /// Split any shared vertices so that this face may be moved independently of the main object. /// /// The source mesh. /// The faces to split from the mesh. /// The faces created forming the detached face group. public static List DetachFaces(this ProBuilderMesh mesh, IEnumerable faces) { return DetachFaces(mesh, faces, true); } /// /// Split any shared vertices so that this face may be moved independently of the main object. /// /// The source mesh. /// The faces to split from the mesh. /// Whether or not to delete the faces on the source geometry which were detached. /// The faces created forming the detached face group. public static List DetachFaces(this ProBuilderMesh mesh, IEnumerable faces, bool deleteSourceFaces) { if (mesh == null) throw new System.ArgumentNullException("mesh"); if (faces == null) throw new System.ArgumentNullException("faces"); List vertices = new List(mesh.GetVertices()); int sharedIndexOffset = mesh.sharedVerticesInternal.Length; var lookup = mesh.sharedVertexLookup; List detached = new List(); foreach (Face face in faces) { FaceRebuildData data = new FaceRebuildData(); data.vertices = new List(); data.sharedIndexes = new List(); data.face = new Face(face); Dictionary match = new Dictionary(); int[] indexes = new int[face.indexesInternal.Length]; for (int i = 0; i < face.indexesInternal.Length; i++) { int local; if (match.TryGetValue(face.indexesInternal[i], out local)) { indexes[i] = local; } else { local = data.vertices.Count; indexes[i] = local; match.Add(face.indexesInternal[i], local); data.vertices.Add(vertices[face.indexesInternal[i]]); data.sharedIndexes.Add(lookup[face.indexesInternal[i]] + sharedIndexOffset); } } data.face.indexesInternal = indexes.ToArray(); detached.Add(data); } FaceRebuildData.Apply(detached, mesh, vertices); if (deleteSourceFaces) { mesh.DeleteFaces(faces); } mesh.ToMesh(); return detached.Select(x => x.face).ToList(); } /// /// Extrude each face in faces individually along it's normal by distance. /// /// /// /// /// static Face[] ExtrudePerFace(ProBuilderMesh pb, IEnumerable faces, float distance) { Face[] faceArray = faces as Face[] ?? faces.ToArray(); if (!faceArray.Any()) return null; List vertices = new List(pb.GetVertices()); int sharedIndexMax = pb.sharedVerticesInternal.Length; int sharedIndexOffset = 0; int faceIndex = 0; Dictionary lookup = pb.sharedVertexLookup; Dictionary lookupUV = pb.sharedTextureLookup; Dictionary used = new Dictionary(); Face[] newFaces = new Face[faceArray.Sum(x => x.edges.Count)]; foreach (Face face in faceArray) { face.smoothingGroup = Smoothing.smoothingGroupNone; face.textureGroup = -1; Vector3 delta = Math.Normal(pb, face) * distance; Edge[] edges = face.edgesInternal; used.Clear(); for (int i = 0; i < edges.Length; i++) { int vc = vertices.Count; int x = edges[i].a, y = edges[i].b; if (!used.ContainsKey(x)) { used.Add(x, lookup[x]); lookup[x] = sharedIndexMax + (sharedIndexOffset++); } if (!used.ContainsKey(y)) { used.Add(y, lookup[y]); lookup[y] = sharedIndexMax + (sharedIndexOffset++); } lookup.Add(vc + 0, used[x]); lookup.Add(vc + 1, used[y]); lookup.Add(vc + 2, lookup[x]); lookup.Add(vc + 3, lookup[y]); Vertex xx = new Vertex(vertices[x]), yy = new Vertex(vertices[y]); xx.position += delta; yy.position += delta; vertices.Add(new Vertex(vertices[x])); vertices.Add(new Vertex(vertices[y])); vertices.Add(xx); vertices.Add(yy); Face bridge = new Face( new int[6] { vc + 0, vc + 1, vc + 2, vc + 1, vc + 3, vc + 2 }, face.submeshIndex, new AutoUnwrapSettings(face.uv), face.smoothingGroup, -1, -1, false ); newFaces[faceIndex++] = bridge; } for (int i = 0; i < face.distinctIndexesInternal.Length; i++) { vertices[face.distinctIndexesInternal[i]].position += delta; // Break any UV shared connections if (lookupUV != null && lookupUV.ContainsKey(face.distinctIndexesInternal[i])) lookupUV.Remove(face.distinctIndexesInternal[i]); } } pb.SetVertices(vertices); var fc = pb.faceCount; var nc = newFaces.Length; var appended = new Face[fc + nc]; Array.Copy(pb.facesInternal, 0, appended, 0, fc); Array.Copy(newFaces, 0, appended, fc, nc); pb.faces = appended; pb.SetSharedVertices(lookup); pb.SetSharedTextures(lookupUV); return newFaces; } /// /// Extrude faces as groups. /// /// /// /// /// /// static Face[] ExtrudeAsGroups(ProBuilderMesh mesh, IEnumerable faces, bool compensateAngleVertexDistance, float distance) { if (faces == null || !faces.Any()) return null; List vertices = new List(mesh.GetVertices()); int sharedIndexMax = mesh.sharedVerticesInternal.Length; int sharedIndexOffset = 0; Dictionary lookup = mesh.sharedVertexLookup; Dictionary lookupUV = mesh.sharedTextureLookup; List newFaces = new List(); // old triangle index -> old shared index Dictionary oldSharedMap = new Dictionary(); // old shared index -> new shared index Dictionary newSharedMap = new Dictionary(); // bridge face extruded edges, maps vertex index to new extruded vertex position Dictionary delayPosition = new Dictionary(); // used to average the direction of vertices shared by perimeter edges // key[shared index], value[normal count, normal sum] Dictionary>> extrudeMap = new Dictionary>>(); List wings = WingedEdge.GetWingedEdges(mesh, faces, true); List> groups = GetFaceGroups(wings); foreach (HashSet group in groups) { Dictionary perimeter = GetPerimeterEdges(group, lookup); newSharedMap.Clear(); oldSharedMap.Clear(); foreach (var edgeAndFace in perimeter) { EdgeLookup edge = edgeAndFace.Key; Face face = edgeAndFace.Value; int vc = vertices.Count; int x = edge.local.a, y = edge.local.b; if (!oldSharedMap.ContainsKey(x)) { oldSharedMap.Add(x, lookup[x]); int newSharedIndex = -1; if (newSharedMap.TryGetValue(lookup[x], out newSharedIndex)) { lookup[x] = newSharedIndex; } else { newSharedIndex = sharedIndexMax + (sharedIndexOffset++); newSharedMap.Add(lookup[x], newSharedIndex); lookup[x] = newSharedIndex; } } if (!oldSharedMap.ContainsKey(y)) { oldSharedMap.Add(y, lookup[y]); int newSharedIndex = -1; if (newSharedMap.TryGetValue(lookup[y], out newSharedIndex)) { lookup[y] = newSharedIndex; } else { newSharedIndex = sharedIndexMax + (sharedIndexOffset++); newSharedMap.Add(lookup[y], newSharedIndex); lookup[y] = newSharedIndex; } } lookup.Add(vc + 0, oldSharedMap[x]); lookup.Add(vc + 1, oldSharedMap[y]); lookup.Add(vc + 2, lookup[x]); lookup.Add(vc + 3, lookup[y]); delayPosition.Add(vc + 2, x); delayPosition.Add(vc + 3, y); vertices.Add(new Vertex(vertices[x])); vertices.Add(new Vertex(vertices[y])); // extruded edge will be positioned later vertices.Add(null); vertices.Add(null); Face bridge = new Face( new int[6] { vc + 0, vc + 1, vc + 2, vc + 1, vc + 3, vc + 2 }, face.submeshIndex, new AutoUnwrapSettings(face.uv), Smoothing.smoothingGroupNone, -1, -1, false ); newFaces.Add(bridge); } foreach (Face face in group) { // @todo keep together if possible face.textureGroup = -1; Vector3 normal = Math.Normal(mesh, face); for (int i = 0; i < face.distinctIndexesInternal.Length; i++) { int idx = face.distinctIndexesInternal[i]; // If this vertex is on the perimeter but not part of a perimeter edge // move the sharedIndex to match it's new value. if (!oldSharedMap.ContainsKey(idx) && newSharedMap.ContainsKey(lookup[idx])) lookup[idx] = newSharedMap[lookup[idx]]; int com = lookup[idx]; // Break any UV shared connections if (lookupUV != null && lookupUV.ContainsKey(face.distinctIndexesInternal[i])) lookupUV.Remove(face.distinctIndexesInternal[i]); // add the normal to the list of normals for this shared vertex SimpleTuple> dir; if (extrudeMap.TryGetValue(com, out dir)) { dir.item1 += normal; dir.item3.Add(idx); extrudeMap[com] = dir; } else { extrudeMap.Add(com, new SimpleTuple>(normal, normal, new List() { idx })); } } } } foreach (var kvp in extrudeMap) { Vector3 direction = (kvp.Value.item1 / kvp.Value.item3.Count); direction.Normalize(); // If extruding by face normal extend vertices on seams by the hypotenuse float modifier = compensateAngleVertexDistance ? Math.Secant(Vector3.Angle(direction, kvp.Value.item2) * Mathf.Deg2Rad) : 1f; direction.x *= distance * modifier; direction.y *= distance * modifier; direction.z *= distance * modifier; foreach (int i in kvp.Value.item3) { vertices[i].position += direction; } } foreach (var kvp in delayPosition) vertices[kvp.Key] = new Vertex(vertices[kvp.Value]); mesh.SetVertices(vertices); var fc = mesh.faceCount; var nc = newFaces.Count; var appended = new Face[fc + nc]; Array.Copy(mesh.facesInternal, 0, appended, 0, fc); for (int i = fc, c = fc + nc; i < c; i++) appended[i] = newFaces[i - fc]; mesh.faces = appended; mesh.SetSharedVertices(lookup); mesh.SetSharedTextures(lookupUV); return newFaces.ToArray(); } static List> GetFaceGroups(List wings) { HashSet used = new HashSet(); List> groups = new List>(); foreach (WingedEdge wing in wings) { if (used.Add(wing.face)) { HashSet group = new HashSet() { wing.face }; ElementSelection.Flood(wing, group); foreach (Face f in group) used.Add(f); groups.Add(group); } } return groups; } static Dictionary GetPerimeterEdges(HashSet faces, Dictionary lookup) { Dictionary perimeter = new Dictionary(); HashSet used = new HashSet(); foreach (Face face in faces) { foreach (Edge edge in face.edgesInternal) { EdgeLookup e = new EdgeLookup(lookup[edge.a], lookup[edge.b], edge.a, edge.b); if (!used.Add(e)) { if (perimeter.ContainsKey(e)) perimeter.Remove(e); } else { perimeter.Add(e, face); } } } return perimeter; } } }