CombineChildren.cs 4.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. using UnityEngine;
  2. using System.Collections;
  3. /*
  4. Attach this script as a parent to some game objects. The script will then combine the meshes at startup.
  5. This is useful as a performance optimization since it is faster to render one big mesh than many small meshes. See the docs on graphics performance optimization for more info.
  6. Different materials will cause multiple meshes to be created, thus it is useful to share as many textures/material as you can.
  7. */
  8. [AddComponentMenu("Mesh/Combine Children")]
  9. public class CombineChildren : MonoBehaviour
  10. {
  11. /// Usually rendering with triangle strips is faster.
  12. /// However when combining objects with very low triangle counts, it can be faster to use triangles.
  13. /// Best is to try out which value is faster in practice.
  14. public bool generateTriangleStrips = true;
  15. /// This option has a far longer preprocessing time at startup but leads to better runtime performance.
  16. void Start()
  17. {
  18. Component[] filters = GetComponentsInChildren(typeof(MeshFilter));
  19. Matrix4x4 myTransform = transform.worldToLocalMatrix;
  20. Hashtable materialToMesh = new Hashtable();
  21. for (int i = 0; i < filters.Length; i++)
  22. {
  23. MeshFilter filter = (MeshFilter)filters[i];
  24. Renderer curRenderer = filters[i].GetComponent<Renderer>();
  25. MeshCombineUtility.MeshInstance instance = new MeshCombineUtility.MeshInstance();
  26. instance.mesh = filter.sharedMesh;
  27. if (curRenderer != null && curRenderer.enabled && instance.mesh != null)
  28. {
  29. instance.transform = myTransform * filter.transform.localToWorldMatrix;
  30. Material[] materials = curRenderer.sharedMaterials;
  31. for (int m = 0; m < materials.Length; m++)
  32. {
  33. instance.subMeshIndex = System.Math.Min(m, instance.mesh.subMeshCount - 1);
  34. ArrayList objects = (ArrayList)materialToMesh[materials[m]];
  35. if (objects != null)
  36. {
  37. objects.Add(instance);
  38. }
  39. else
  40. {
  41. objects = new ArrayList();
  42. objects.Add(instance);
  43. materialToMesh.Add(materials[m], objects);
  44. }
  45. }
  46. curRenderer.enabled = false;
  47. }
  48. }
  49. foreach (DictionaryEntry de in materialToMesh)
  50. {
  51. ArrayList elements = (ArrayList)de.Value;
  52. MeshCombineUtility.MeshInstance[] instances = (MeshCombineUtility.MeshInstance[])elements.ToArray(typeof(MeshCombineUtility.MeshInstance));
  53. // We have a maximum of one material, so just attach the mesh to our own game object
  54. if (materialToMesh.Count == 1)
  55. {
  56. // Make sure we have a mesh filter & renderer
  57. if (GetComponent(typeof(MeshFilter)) == null)
  58. gameObject.AddComponent(typeof(MeshFilter));
  59. if (!GetComponent<MeshRenderer>())
  60. gameObject.AddComponent<MeshRenderer>();
  61. MeshFilter filter = (MeshFilter)GetComponent(typeof(MeshFilter));
  62. filter.mesh = MeshCombineUtility.Combine(instances, generateTriangleStrips);
  63. GetComponent<Renderer>().material = (Material)de.Key;
  64. GetComponent<Renderer>().enabled = true;
  65. }
  66. // We have multiple materials to take care of, build one mesh / gameobject for each material
  67. // and parent it to this object
  68. else
  69. {
  70. GameObject go = new GameObject("Combined mesh");
  71. go.transform.parent = transform;
  72. go.transform.localScale = Vector3.one;
  73. go.transform.localRotation = Quaternion.identity;
  74. go.transform.localPosition = Vector3.zero;
  75. go.AddComponent(typeof(MeshFilter));
  76. go.AddComponent<MeshRenderer>();
  77. go.GetComponent<Renderer>().material = (Material)de.Key;
  78. MeshFilter filter = (MeshFilter)go.GetComponent(typeof(MeshFilter));
  79. filter.mesh = MeshCombineUtility.Combine(instances, generateTriangleStrips);
  80. }
  81. }
  82. }
  83. }