765 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			765 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|  | //---------------------------------------------- | ||
|  | //            MeshBaker | ||
|  | // Copyright © 2011-2012 Ian Deane | ||
|  | //---------------------------------------------- | ||
|  | using UnityEngine; | ||
|  | using System.Collections; | ||
|  | using System.Collections.Specialized; | ||
|  | using System; | ||
|  | using System.Collections.Generic; | ||
|  | using System.Text; | ||
|  | using System.IO; | ||
|  | using DigitalOpus.MB.Core; | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | /// <summary> | ||
|  | /// Component that handles baking materials into a combined material. | ||
|  | ///  | ||
|  | /// The result of the material baking process is a MB2_TextureBakeResults object, which  | ||
|  | /// becomes the input for the mesh baking. | ||
|  | ///  | ||
|  | /// This class uses the MB_TextureCombiner to do the combining. | ||
|  | ///  | ||
|  | /// This class is a Component (MonoBehavior) so it is serialized and found using GetComponent. If | ||
|  | /// you want to access the texture baking functionality without creating a Component then use MB_TextureCombiner | ||
|  | /// directly. | ||
|  | /// </summary> | ||
|  | public class MB3_TextureBaker : MB3_MeshBakerRoot | ||
|  | { | ||
|  |     public MB2_LogLevel LOG_LEVEL = MB2_LogLevel.info; | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected MB2_TextureBakeResults _textureBakeResults; | ||
|  |     public override MB2_TextureBakeResults textureBakeResults | ||
|  |     { | ||
|  |         get { return _textureBakeResults; } | ||
|  |         set { _textureBakeResults = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected int _atlasPadding = 1; | ||
|  |     public virtual int atlasPadding | ||
|  |     { | ||
|  |         get { return _atlasPadding; } | ||
|  |         set { _atlasPadding = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected int _maxAtlasSize = 4096; | ||
|  |     public virtual int maxAtlasSize | ||
|  |     { | ||
|  |         get { return _maxAtlasSize; } | ||
|  |         set { _maxAtlasSize = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _useMaxAtlasWidthOverride = false; | ||
|  |     public virtual bool useMaxAtlasWidthOverride | ||
|  |     { | ||
|  |         get { return _useMaxAtlasWidthOverride; } | ||
|  |         set { _useMaxAtlasWidthOverride = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected int _maxAtlasWidthOverride = 4096; | ||
|  |     public virtual int maxAtlasWidthOverride | ||
|  |     { | ||
|  |         get { return _maxAtlasWidthOverride; } | ||
|  |         set { _maxAtlasWidthOverride = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _useMaxAtlasHeightOverride = false; | ||
|  |     public virtual bool useMaxAtlasHeightOverride | ||
|  |     { | ||
|  |         get { return _useMaxAtlasHeightOverride; } | ||
|  |         set { _useMaxAtlasHeightOverride = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected int _maxAtlasHeightOverride = 4096; | ||
|  |     public virtual int maxAtlasHeightOverride | ||
|  |     { | ||
|  |         get { return _maxAtlasHeightOverride; } | ||
|  |         set { _maxAtlasHeightOverride = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _resizePowerOfTwoTextures = false; | ||
|  |     public virtual bool resizePowerOfTwoTextures | ||
|  |     { | ||
|  |         get { return _resizePowerOfTwoTextures; } | ||
|  |         set { _resizePowerOfTwoTextures = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _fixOutOfBoundsUVs = false; //is considerMeshUVs but can't change because it would break all existing bakers | ||
|  |     public virtual bool fixOutOfBoundsUVs | ||
|  |     { | ||
|  |         get { return _fixOutOfBoundsUVs; } | ||
|  |         set { _fixOutOfBoundsUVs = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected int _maxTilingBakeSize = 1024; | ||
|  |     public virtual int maxTilingBakeSize | ||
|  |     { | ||
|  |         get { return _maxTilingBakeSize; } | ||
|  |         set { _maxTilingBakeSize = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected MB2_PackingAlgorithmEnum _packingAlgorithm = MB2_PackingAlgorithmEnum.MeshBakerTexturePacker; | ||
|  |     public virtual MB2_PackingAlgorithmEnum packingAlgorithm | ||
|  |     { | ||
|  |         get { return _packingAlgorithm; } | ||
|  |         set { _packingAlgorithm = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected int _layerTexturePackerFastMesh = -1; | ||
|  |     public virtual int layerForTexturePackerFastMesh | ||
|  |     { | ||
|  |         get { return _layerTexturePackerFastMesh; } | ||
|  |         set { _layerTexturePackerFastMesh = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _meshBakerTexturePackerForcePowerOfTwo = true; | ||
|  |     public bool meshBakerTexturePackerForcePowerOfTwo | ||
|  |     { | ||
|  |         get { return _meshBakerTexturePackerForcePowerOfTwo; } | ||
|  |         set { _meshBakerTexturePackerForcePowerOfTwo = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected List<ShaderTextureProperty> _customShaderProperties = new List<ShaderTextureProperty>(); | ||
|  |     public virtual List<ShaderTextureProperty> customShaderProperties | ||
|  |     { | ||
|  |         get { return _customShaderProperties; } | ||
|  |         set { _customShaderProperties = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |      | ||
|  |     [SerializeField] | ||
|  |     protected List<string> _texturePropNamesToIgnore = new List<string>(); | ||
|  |     public virtual List<string> texturePropNamesToIgnore | ||
|  |     { | ||
|  |         get { return _texturePropNamesToIgnore; } | ||
|  |         set { _texturePropNamesToIgnore = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     //this is depricated | ||
|  |     [SerializeField] | ||
|  |     protected List<string> _customShaderPropNames_Depricated = new List<string>(); | ||
|  |     public virtual List<string> customShaderPropNames | ||
|  |     { | ||
|  |         get { return _customShaderPropNames_Depricated; } | ||
|  |         set { _customShaderPropNames_Depricated = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected MB2_TextureBakeResults.ResultType _resultType; | ||
|  |     public virtual MB2_TextureBakeResults.ResultType resultType | ||
|  |     { | ||
|  |         get { return _resultType; } | ||
|  |         set { _resultType = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _doMultiMaterial; | ||
|  |     public virtual bool doMultiMaterial | ||
|  |     { | ||
|  |         get { return _doMultiMaterial; } | ||
|  |         set { _doMultiMaterial = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _doMultiMaterialSplitAtlasesIfTooBig = true; | ||
|  |     public virtual bool doMultiMaterialSplitAtlasesIfTooBig | ||
|  |     { | ||
|  |         get { return _doMultiMaterialSplitAtlasesIfTooBig; } | ||
|  |         set { _doMultiMaterialSplitAtlasesIfTooBig = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _doMultiMaterialSplitAtlasesIfOBUVs = true; | ||
|  |     public virtual bool doMultiMaterialSplitAtlasesIfOBUVs | ||
|  |     { | ||
|  |         get { return _doMultiMaterialSplitAtlasesIfOBUVs; } | ||
|  |         set { _doMultiMaterialSplitAtlasesIfOBUVs = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected Material _resultMaterial; | ||
|  |     public virtual Material resultMaterial | ||
|  |     { | ||
|  |         get { return _resultMaterial; } | ||
|  |         set { _resultMaterial = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _considerNonTextureProperties = false; | ||
|  |     public bool considerNonTextureProperties | ||
|  |     { | ||
|  |         get { return _considerNonTextureProperties; } | ||
|  |         set { _considerNonTextureProperties = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     [SerializeField] | ||
|  |     protected bool _doSuggestTreatment = true; | ||
|  |     public bool doSuggestTreatment | ||
|  |     { | ||
|  |         get { return _doSuggestTreatment; } | ||
|  |         set { _doSuggestTreatment = value; } | ||
|  |     } | ||
|  | 
 | ||
|  |     private MB3_TextureCombiner.CreateAtlasesCoroutineResult _coroutineResult; | ||
|  |     public MB3_TextureCombiner.CreateAtlasesCoroutineResult CoroutineResult | ||
|  |     { | ||
|  |         get | ||
|  |         { | ||
|  |             return _coroutineResult; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     public MB_MultiMaterial[] resultMaterials = new MB_MultiMaterial[0]; | ||
|  | 
 | ||
|  |     public MB_MultiMaterialTexArray[] resultMaterialsTexArray = new MB_MultiMaterialTexArray[0]; | ||
|  | 
 | ||
|  |     public MB_TextureArrayFormatSet[] textureArrayOutputFormats; | ||
|  | 
 | ||
|  |     public List<GameObject> objsToMesh; //todo make this Renderer | ||
|  | 
 | ||
|  |     public override List<GameObject> GetObjectsToCombine() | ||
|  |     { | ||
|  |         if (objsToMesh == null) objsToMesh = new List<GameObject>(); | ||
|  |         return objsToMesh; | ||
|  |     } | ||
|  | 
 | ||
|  |     [ContextMenu("Purge Objects to Combine of null references")] | ||
|  |     public override void PurgeNullsFromObjectsToCombine() | ||
|  |     { | ||
|  |         if (objsToMesh == null) | ||
|  |         { | ||
|  |             objsToMesh = new List<GameObject>(); | ||
|  |         } | ||
|  |         Debug.Log(string.Format("Purged {0} null references from objects to combine list.", objsToMesh.RemoveAll(obj => obj == null))); | ||
|  |     } | ||
|  | 
 | ||
|  |     public MB_AtlasesAndRects[] CreateAtlases() | ||
|  |     { | ||
|  |         return CreateAtlases(null, false, null); | ||
|  |     } | ||
|  | 
 | ||
|  |     public delegate void OnCombinedTexturesCoroutineSuccess(); | ||
|  |     public delegate void OnCombinedTexturesCoroutineFail(); | ||
|  |     public OnCombinedTexturesCoroutineSuccess onBuiltAtlasesSuccess; | ||
|  |     public OnCombinedTexturesCoroutineFail onBuiltAtlasesFail; | ||
|  |     public MB_AtlasesAndRects[] OnCombinedTexturesCoroutineAtlasesAndRects; | ||
|  | 
 | ||
|  |     public IEnumerator CreateAtlasesCoroutine(ProgressUpdateDelegate progressInfo, MB3_TextureCombiner.CreateAtlasesCoroutineResult coroutineResult, bool saveAtlasesAsAssets = false, MB2_EditorMethodsInterface editorMethods = null, float maxTimePerFrame = .01f) | ||
|  |     { | ||
|  |         yield return _CreateAtlasesCoroutine(progressInfo, coroutineResult, saveAtlasesAsAssets, editorMethods, maxTimePerFrame); | ||
|  | 
 | ||
|  |         if (coroutineResult.success && onBuiltAtlasesSuccess != null) | ||
|  |         { | ||
|  |             onBuiltAtlasesSuccess(); | ||
|  |         } | ||
|  | 
 | ||
|  |         if (!coroutineResult.success && onBuiltAtlasesFail != null) | ||
|  |         { | ||
|  |             onBuiltAtlasesFail(); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     private IEnumerator _CreateAtlasesCoroutineAtlases(MB3_TextureCombiner combiner, ProgressUpdateDelegate progressInfo, MB3_TextureCombiner.CreateAtlasesCoroutineResult coroutineResult, bool saveAtlasesAsAssets = false, MB2_EditorMethodsInterface editorMethods = null, float maxTimePerFrame = .01f) | ||
|  |     { | ||
|  |         int numResults = 1; | ||
|  |         if (_doMultiMaterial) | ||
|  |         { | ||
|  |             numResults = resultMaterials.Length; | ||
|  |         } | ||
|  | 
 | ||
|  |         OnCombinedTexturesCoroutineAtlasesAndRects = new MB_AtlasesAndRects[numResults]; | ||
|  |         for (int i = 0; i < OnCombinedTexturesCoroutineAtlasesAndRects.Length; i++) | ||
|  |         { | ||
|  |             OnCombinedTexturesCoroutineAtlasesAndRects[i] = new MB_AtlasesAndRects(); | ||
|  |         } | ||
|  | 
 | ||
|  |         //Do the material combining. | ||
|  |         for (int i = 0; i < OnCombinedTexturesCoroutineAtlasesAndRects.Length; i++) | ||
|  |         { | ||
|  |             Material resMatToPass = null; | ||
|  |             List<Material> sourceMats = null; | ||
|  |             if (_doMultiMaterial) | ||
|  |             { | ||
|  |                 sourceMats = resultMaterials[i].sourceMaterials; | ||
|  |                 resMatToPass = resultMaterials[i].combinedMaterial; | ||
|  |                 combiner.fixOutOfBoundsUVs = resultMaterials[i].considerMeshUVs; | ||
|  |             } | ||
|  |             else | ||
|  |             { | ||
|  |                 resMatToPass = _resultMaterial; | ||
|  |             } | ||
|  | 
 | ||
|  |             MB3_TextureCombiner.CombineTexturesIntoAtlasesCoroutineResult coroutineResult2 = new MB3_TextureCombiner.CombineTexturesIntoAtlasesCoroutineResult(); | ||
|  |             yield return combiner.CombineTexturesIntoAtlasesCoroutine(progressInfo, OnCombinedTexturesCoroutineAtlasesAndRects[i], resMatToPass, objsToMesh, sourceMats, texturePropNamesToIgnore, editorMethods, coroutineResult2, maxTimePerFrame, | ||
|  |                 onlyPackRects: false, splitAtlasWhenPackingIfTooBig: false); | ||
|  |             coroutineResult.success = coroutineResult2.success; | ||
|  |             if (!coroutineResult.success) | ||
|  |             { | ||
|  |                 coroutineResult.isFinished = true; | ||
|  |                 yield break; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         unpackMat2RectMap(OnCombinedTexturesCoroutineAtlasesAndRects); | ||
|  |         if (coroutineResult.success && editorMethods != null) editorMethods.GetMaterialPrimaryKeysIfAddressables(textureBakeResults); | ||
|  |         //Save the results | ||
|  |         textureBakeResults.resultType = MB2_TextureBakeResults.ResultType.atlas; | ||
|  |         textureBakeResults.resultMaterialsTexArray = new MB_MultiMaterialTexArray[0]; | ||
|  |         textureBakeResults.doMultiMaterial = _doMultiMaterial; | ||
|  |         if (_doMultiMaterial) | ||
|  |         { | ||
|  |             textureBakeResults.resultMaterials = resultMaterials; | ||
|  |         } | ||
|  |         else | ||
|  |         { | ||
|  |             MB_MultiMaterial[] resMats = new MB_MultiMaterial[1]; | ||
|  |             resMats[0] = new MB_MultiMaterial(); | ||
|  |             resMats[0].combinedMaterial = _resultMaterial; | ||
|  |             resMats[0].considerMeshUVs = _fixOutOfBoundsUVs; | ||
|  |             resMats[0].sourceMaterials = new List<Material>(); | ||
|  |             for (int i = 0; i < textureBakeResults.materialsAndUVRects.Length; i++) | ||
|  |             { | ||
|  |                 resMats[0].sourceMaterials.Add(textureBakeResults.materialsAndUVRects[i].material); | ||
|  |             } | ||
|  |             textureBakeResults.resultMaterials = resMats; | ||
|  |         } | ||
|  | 
 | ||
|  |         if (LOG_LEVEL >= MB2_LogLevel.info) Debug.Log("Created Atlases"); | ||
|  |     } | ||
|  | 
 | ||
|  |     internal IEnumerator _CreateAtlasesCoroutineTextureArray(MB3_TextureCombiner combiner, ProgressUpdateDelegate progressInfo, MB3_TextureCombiner.CreateAtlasesCoroutineResult coroutineResult, bool saveAtlasesAsAssets = false, MB2_EditorMethodsInterface editorMethods = null, float maxTimePerFrame = .01f) | ||
|  |     { | ||
|  |         MB_TextureArrayResultMaterial[] bakedMatsAndSlices = null; | ||
|  | 
 | ||
|  |         // Validate the formats | ||
|  |         if (textureArrayOutputFormats == null || textureArrayOutputFormats.Length == 0) | ||
|  |         { | ||
|  |             Debug.LogError("No Texture Array Output Formats. There must be at least one entry."); | ||
|  |             coroutineResult.isFinished = true; | ||
|  |             yield break; | ||
|  |         } | ||
|  | 
 | ||
|  |         for (int i = 0; i < textureArrayOutputFormats.Length; i++) | ||
|  |         { | ||
|  |             if (!textureArrayOutputFormats[i].ValidateTextureImporterFormatsExistsForTextureFormats(editorMethods, i)) | ||
|  |             { | ||
|  |                 Debug.LogError("Could not map the selected texture format to a Texture Importer Format. Safest options are ARGB32, or RGB24."); | ||
|  |                 coroutineResult.isFinished = true; | ||
|  |                 yield break; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         for (int resMatIdx = 0; resMatIdx < resultMaterialsTexArray.Length; resMatIdx++) | ||
|  |         { | ||
|  |             MB_MultiMaterialTexArray textureArraySliceConfig = resultMaterialsTexArray[resMatIdx]; | ||
|  |             if (textureArraySliceConfig.combinedMaterial == null) | ||
|  |             { | ||
|  |                 Debug.LogError("Material is null for Texture Array Slice Configuration: " + resMatIdx + "."); | ||
|  |                 coroutineResult.isFinished = true; | ||
|  |                 yield break; | ||
|  |             } | ||
|  | 
 | ||
|  | 
 | ||
|  |             List<MB_TexArraySlice> slices = textureArraySliceConfig.slices; | ||
|  |             for (int sliceIdx = 0; sliceIdx < slices.Count; sliceIdx++) | ||
|  |             { | ||
|  |                 for (int srcMatIdx = 0; srcMatIdx < slices[sliceIdx].sourceMaterials.Count; srcMatIdx++) | ||
|  |                 { | ||
|  |                     MB_TexArraySliceRendererMatPair sourceMat = slices[sliceIdx].sourceMaterials[srcMatIdx]; | ||
|  |                     if (sourceMat.sourceMaterial == null) | ||
|  |                     { | ||
|  |                         Debug.LogError("Source material is null for Texture Array Slice Configuration: " + resMatIdx + " slice: " + sliceIdx); | ||
|  |                         coroutineResult.isFinished = true; | ||
|  |                         yield break; | ||
|  |                     } | ||
|  | 
 | ||
|  |                     if (slices[sliceIdx].considerMeshUVs) | ||
|  |                     { | ||
|  |                         if (sourceMat.renderer == null) | ||
|  |                         { | ||
|  |                             Debug.LogError("Renderer is null for Texture Array Slice Configuration: " + resMatIdx + " slice: " + sliceIdx + ". If considerUVs is enabled then a renderer must be supplied for each source material. The same source material can be used multiple times."); | ||
|  |                             coroutineResult.isFinished = true; | ||
|  |                             yield break; | ||
|  |                         } | ||
|  |                     } | ||
|  |                     else | ||
|  |                     { | ||
|  |                         // TODO check for duplicate source mats. | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         // initialize structure to store results. For texture arrays the structure is two layers deep. | ||
|  |         // First layer is resultMaterial / submesh (each result material can use a different shader) | ||
|  |         // Second layer is a set of TextureArrays for the TextureProperties on that result material. | ||
|  |         int numResultMats = resultMaterialsTexArray.Length; | ||
|  |         bakedMatsAndSlices = new MB_TextureArrayResultMaterial[numResultMats]; | ||
|  |         for (int resMatIdx = 0; resMatIdx < bakedMatsAndSlices.Length; resMatIdx++) | ||
|  |         { | ||
|  |             bakedMatsAndSlices[resMatIdx] = new MB_TextureArrayResultMaterial(); | ||
|  |             int numSlices = resultMaterialsTexArray[resMatIdx].slices.Count; | ||
|  |             MB_AtlasesAndRects[] slices = bakedMatsAndSlices[resMatIdx].slices = new MB_AtlasesAndRects[numSlices]; | ||
|  |             for (int j = 0; j < numSlices; j++) slices[j] = new MB_AtlasesAndRects(); | ||
|  |         } | ||
|  | 
 | ||
|  |         // Some of the slices will be atlases (more than one atlas per slice). | ||
|  |         // Do the material combining for these. First loop over the result materials (1 per submeshes). | ||
|  |         for (int resMatIdx = 0; resMatIdx < bakedMatsAndSlices.Length; resMatIdx++) | ||
|  |         { | ||
|  |             yield return MB_TextureArrays._CreateAtlasesCoroutineSingleResultMaterial(resMatIdx, bakedMatsAndSlices[resMatIdx], resultMaterialsTexArray[resMatIdx], | ||
|  |                 objsToMesh, | ||
|  |                 combiner,  | ||
|  |                 textureArrayOutputFormats, | ||
|  |                 resultMaterialsTexArray, | ||
|  |                 customShaderProperties, | ||
|  |                 texturePropNamesToIgnore, | ||
|  |                 progressInfo, coroutineResult, saveAtlasesAsAssets, editorMethods, maxTimePerFrame); | ||
|  |             if (!coroutineResult.success) yield break; | ||
|  |         } | ||
|  | 
 | ||
|  |         if (coroutineResult.success) | ||
|  |         { | ||
|  |             // Save the results into the TextureBakeResults. | ||
|  |             unpackMat2RectMap(bakedMatsAndSlices); | ||
|  |             if (editorMethods != null) editorMethods.GetMaterialPrimaryKeysIfAddressables(textureBakeResults); | ||
|  |             textureBakeResults.resultType = MB2_TextureBakeResults.ResultType.textureArray; | ||
|  |             textureBakeResults.resultMaterials = new MB_MultiMaterial[0]; | ||
|  |             textureBakeResults.resultMaterialsTexArray = resultMaterialsTexArray; | ||
|  |             if (LOG_LEVEL >= MB2_LogLevel.info) Debug.Log("Created Texture2DArrays"); | ||
|  |         } | ||
|  |         else | ||
|  |         { | ||
|  |             if (LOG_LEVEL >= MB2_LogLevel.info) Debug.Log("Failed to create Texture2DArrays"); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     private IEnumerator _CreateAtlasesCoroutine(ProgressUpdateDelegate progressInfo, MB3_TextureCombiner.CreateAtlasesCoroutineResult coroutineResult, bool saveAtlasesAsAssets = false, MB2_EditorMethodsInterface editorMethods = null, float maxTimePerFrame = .01f) | ||
|  |     { | ||
|  |         MBVersionConcrete mbv = new MBVersionConcrete(); | ||
|  |         if (!MB3_TextureCombiner._RunCorutineWithoutPauseIsRunning && (mbv.GetMajorVersion() < 5 || (mbv.GetMajorVersion() == 5 && mbv.GetMinorVersion() < 3))) | ||
|  |         { | ||
|  |             Debug.LogError("Running the texture combiner as a coroutine only works in Unity 5.3 and higher"); | ||
|  |             coroutineResult.success = false; | ||
|  |             yield break; | ||
|  |         } | ||
|  |         this.OnCombinedTexturesCoroutineAtlasesAndRects = null; | ||
|  | 
 | ||
|  |         if (maxTimePerFrame <= 0f) | ||
|  |         { | ||
|  |             Debug.LogError("maxTimePerFrame must be a value greater than zero"); | ||
|  |             coroutineResult.isFinished = true; | ||
|  |             yield break; | ||
|  |         } | ||
|  |         MB2_ValidationLevel vl = Application.isPlaying ? MB2_ValidationLevel.quick : MB2_ValidationLevel.robust; | ||
|  |         if (!DoCombinedValidate(this, MB_ObjsToCombineTypes.dontCare, null, vl)) | ||
|  |         { | ||
|  |             coroutineResult.isFinished = true; | ||
|  |             yield break; | ||
|  |         } | ||
|  |         if (_doMultiMaterial && !_ValidateResultMaterials()) | ||
|  |         { | ||
|  |             coroutineResult.isFinished = true; | ||
|  |             yield break; | ||
|  |         } | ||
|  |         else if (resultType == MB2_TextureBakeResults.ResultType.textureArray) | ||
|  |         { | ||
|  |             //TODO validate texture arrays. | ||
|  |         } | ||
|  |         else if (!_doMultiMaterial) | ||
|  |         { | ||
|  |             if (_resultMaterial == null) | ||
|  |             { | ||
|  |                 Debug.LogError("Combined Material is null please create and assign a result material."); | ||
|  |                 coroutineResult.isFinished = true; | ||
|  |                 yield break; | ||
|  |             } | ||
|  |             Shader targShader = _resultMaterial.shader; | ||
|  |             for (int i = 0; i < objsToMesh.Count; i++) | ||
|  |             { | ||
|  |                 Material[] ms = MB_Utility.GetGOMaterials(objsToMesh[i]); | ||
|  |                 for (int j = 0; j < ms.Length; j++) | ||
|  |                 { | ||
|  |                     Material m = ms[j]; | ||
|  |                     if (m != null && m.shader != targShader) | ||
|  |                     { | ||
|  |                         Debug.LogWarning("Game object " + objsToMesh[i] + " does not use shader " + targShader + " it may not have the required textures. If not small solid color textures will be generated."); | ||
|  |                     } | ||
|  | 
 | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         MB3_TextureCombiner combiner = CreateAndConfigureTextureCombiner(); | ||
|  |         combiner.saveAtlasesAsAssets = saveAtlasesAsAssets; | ||
|  | 
 | ||
|  |         OnCombinedTexturesCoroutineAtlasesAndRects = null; | ||
|  |         if (resultType == MB2_TextureBakeResults.ResultType.textureArray) | ||
|  |         { | ||
|  |             yield return _CreateAtlasesCoroutineTextureArray(combiner, progressInfo, coroutineResult, saveAtlasesAsAssets, editorMethods, maxTimePerFrame); | ||
|  |             if (!coroutineResult.success) yield break; | ||
|  |         } | ||
|  |         else | ||
|  |         { | ||
|  |             yield return _CreateAtlasesCoroutineAtlases(combiner, progressInfo, coroutineResult, saveAtlasesAsAssets, editorMethods, maxTimePerFrame); | ||
|  |             if (!coroutineResult.success) yield break; | ||
|  |         } | ||
|  | 
 | ||
|  |         //set the texture bake resultAtlasesAndRects on the Mesh Baker component if it exists | ||
|  |         MB3_MeshBakerCommon[] mb = GetComponentsInChildren<MB3_MeshBakerCommon>(); | ||
|  |         for (int i = 0; i < mb.Length; i++) | ||
|  |         { | ||
|  |             mb[i].textureBakeResults = textureBakeResults; | ||
|  |         } | ||
|  | 
 | ||
|  |         coroutineResult.isFinished = true; | ||
|  |     } | ||
|  | 
 | ||
|  |     /// <summary> | ||
|  |     /// Creates the atlases. | ||
|  |     /// </summary> | ||
|  |     /// <returns> | ||
|  |     /// The atlases. | ||
|  |     /// </returns> | ||
|  |     /// <param name='progressInfo'> | ||
|  |     /// Progress info is a delegate function that displays a progress dialog. Can be null | ||
|  |     /// </param> | ||
|  |     /// <param name='saveAtlasesAsAssets'> | ||
|  |     /// if true atlases are saved as assets in the project folder. Othersise they are instances in memory | ||
|  |     /// </param> | ||
|  |     /// <param name='editorMethods'> | ||
|  |     /// Texture format tracker. Contains editor functionality such as save assets. Can be null. | ||
|  |     /// </param> | ||
|  |     public MB_AtlasesAndRects[] CreateAtlases(ProgressUpdateDelegate progressInfo, bool saveAtlasesAsAssets = false, MB2_EditorMethodsInterface editorMethods = null) | ||
|  |     { | ||
|  |         MB_AtlasesAndRects[] mAndAs = null; | ||
|  | 
 | ||
|  |         try | ||
|  |         { | ||
|  |             _coroutineResult = new MB3_TextureCombiner.CreateAtlasesCoroutineResult(); | ||
|  |             MB3_TextureCombiner.RunCorutineWithoutPause(CreateAtlasesCoroutine(progressInfo, _coroutineResult, saveAtlasesAsAssets, editorMethods, 1000f), 0); | ||
|  |             if (_coroutineResult.success && textureBakeResults != null) | ||
|  |             { | ||
|  |                 mAndAs = this.OnCombinedTexturesCoroutineAtlasesAndRects; | ||
|  |             } | ||
|  |         } | ||
|  |         catch (Exception ex) | ||
|  |         { | ||
|  |             Debug.LogError(ex.Message + "\n" + ex.StackTrace.ToString()); | ||
|  |         } | ||
|  |         finally | ||
|  |         { | ||
|  |             if (saveAtlasesAsAssets) | ||
|  |             { //Atlases were saved to project so we don't need these ones | ||
|  |                 if (mAndAs != null) | ||
|  |                 { | ||
|  |                     for (int j = 0; j < mAndAs.Length; j++) | ||
|  |                     { | ||
|  |                         MB_AtlasesAndRects mAndA = mAndAs[j]; | ||
|  |                         if (mAndA != null && mAndA.atlases != null) | ||
|  |                         { | ||
|  |                             for (int i = 0; i < mAndA.atlases.Length; i++) | ||
|  |                             { | ||
|  |                                 if (mAndA.atlases[i] != null) | ||
|  |                                 { | ||
|  |                                     if (editorMethods != null) editorMethods.Destroy(mAndA.atlases[i]); | ||
|  |                                     else MB_Utility.Destroy(mAndA.atlases[i]); | ||
|  |                                 } | ||
|  |                             } | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         return mAndAs; | ||
|  |     } | ||
|  | 
 | ||
|  |     void unpackMat2RectMap(MB_AtlasesAndRects[] rawResults) | ||
|  |     { | ||
|  |         List<MB_MaterialAndUVRect> mss = new List<MB_MaterialAndUVRect>(); | ||
|  |         for (int i = 0; i < rawResults.Length; i++) | ||
|  |         { | ||
|  |             List<MB_MaterialAndUVRect> map = rawResults[i].mat2rect_map; | ||
|  |             if (map != null) | ||
|  |             { | ||
|  |                 for (int j = 0; j < map.Count; j++) | ||
|  |                 { | ||
|  |                     map[j].textureArraySliceIdx = -1; | ||
|  |                     mss.Add(map[j]); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         textureBakeResults.version = MB2_TextureBakeResults.VERSION; | ||
|  |         textureBakeResults.materialsAndUVRects = mss.ToArray(); | ||
|  |     } | ||
|  | 
 | ||
|  |     internal void unpackMat2RectMap(MB_TextureArrayResultMaterial[] rawResults) | ||
|  |     { | ||
|  |         List<MB_MaterialAndUVRect> mss = new List<MB_MaterialAndUVRect>(); | ||
|  |         for (int resMatIdx = 0; resMatIdx < rawResults.Length; resMatIdx++) | ||
|  |         { | ||
|  |             MB_AtlasesAndRects[] slices = rawResults[resMatIdx].slices; | ||
|  |             for (int sliceIdx = 0; sliceIdx < slices.Length; sliceIdx++) | ||
|  |             { | ||
|  |                 List<MB_MaterialAndUVRect> map = slices[sliceIdx].mat2rect_map; | ||
|  |                 if (map != null) | ||
|  |                 { | ||
|  |                     for (int rectIdx = 0; rectIdx < map.Count; rectIdx++) | ||
|  |                     { | ||
|  |                         map[rectIdx].textureArraySliceIdx = sliceIdx; | ||
|  |                         mss.Add(map[rectIdx]); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         textureBakeResults.version = MB2_TextureBakeResults.VERSION; | ||
|  |         textureBakeResults.materialsAndUVRects = mss.ToArray(); | ||
|  |     } | ||
|  | 
 | ||
|  |     public MB3_TextureCombiner CreateAndConfigureTextureCombiner() | ||
|  |     { | ||
|  |         MB3_TextureCombiner combiner = new MB3_TextureCombiner(); | ||
|  |         combiner.LOG_LEVEL = LOG_LEVEL; | ||
|  |         combiner.atlasPadding = _atlasPadding; | ||
|  |         combiner.maxAtlasSize = _maxAtlasSize; | ||
|  |         combiner.maxAtlasHeightOverride = _maxAtlasHeightOverride; | ||
|  |         combiner.maxAtlasWidthOverride = _maxAtlasWidthOverride; | ||
|  |         combiner.useMaxAtlasHeightOverride = _useMaxAtlasHeightOverride; | ||
|  |         combiner.useMaxAtlasWidthOverride = _useMaxAtlasWidthOverride; | ||
|  |         combiner.customShaderPropNames = _customShaderProperties; | ||
|  |         combiner.fixOutOfBoundsUVs = _fixOutOfBoundsUVs; | ||
|  |         combiner.maxTilingBakeSize = _maxTilingBakeSize; | ||
|  |         combiner.packingAlgorithm = _packingAlgorithm; | ||
|  |         combiner.layerTexturePackerFastMesh = _layerTexturePackerFastMesh; | ||
|  |         combiner.resultType = _resultType; | ||
|  |         combiner.meshBakerTexturePackerForcePowerOfTwo = _meshBakerTexturePackerForcePowerOfTwo; | ||
|  |         combiner.resizePowerOfTwoTextures = _resizePowerOfTwoTextures; | ||
|  |         combiner.considerNonTextureProperties = _considerNonTextureProperties; | ||
|  |         return combiner; | ||
|  |     } | ||
|  | 
 | ||
|  |     public static void ConfigureNewMaterialToMatchOld(Material newMat, Material original) | ||
|  |     { | ||
|  |         if (original == null) | ||
|  |         { | ||
|  |             Debug.LogWarning("Original material is null, could not copy properties to " + newMat + ". Setting shader to " + newMat.shader); | ||
|  |             return; | ||
|  |         } | ||
|  |         newMat.shader = original.shader; | ||
|  |         newMat.CopyPropertiesFromMaterial(original); | ||
|  |         ShaderTextureProperty[] texPropertyNames = MB3_TextureCombinerPipeline.shaderTexPropertyNames; | ||
|  |         for (int j = 0; j < texPropertyNames.Length; j++) | ||
|  |         { | ||
|  |             Vector2 scale = Vector2.one; | ||
|  |             Vector2 offset = Vector2.zero; | ||
|  |             if (newMat.HasProperty(texPropertyNames[j].name)) | ||
|  |             { | ||
|  |                 newMat.SetTextureOffset(texPropertyNames[j].name, offset); | ||
|  |                 newMat.SetTextureScale(texPropertyNames[j].name, scale); | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     string PrintSet(HashSet<Material> s) | ||
|  |     { | ||
|  |         StringBuilder sb = new StringBuilder(); | ||
|  |         foreach (Material m in s) | ||
|  |         { | ||
|  |             sb.Append(m + ","); | ||
|  |         } | ||
|  |         return sb.ToString(); | ||
|  |     } | ||
|  | 
 | ||
|  |     bool _ValidateResultMaterials() | ||
|  |     { | ||
|  |         HashSet<Material> allMatsOnObjs = new HashSet<Material>(); | ||
|  |         for (int i = 0; i < objsToMesh.Count; i++) | ||
|  |         { | ||
|  |             if (objsToMesh[i] != null) | ||
|  |             { | ||
|  |                 Material[] ms = MB_Utility.GetGOMaterials(objsToMesh[i]); | ||
|  |                 for (int j = 0; j < ms.Length; j++) | ||
|  |                 { | ||
|  |                     if (ms[j] != null) allMatsOnObjs.Add(ms[j]); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         if (resultMaterials.Length < 1) | ||
|  |         { | ||
|  |             Debug.LogError("Using multiple materials but there are no 'Source Material To Combined Mappings'. You need at least one."); | ||
|  |         } | ||
|  | 
 | ||
|  |         HashSet<Material> allMatsInMapping = new HashSet<Material>(); | ||
|  |         for (int i = 0; i < resultMaterials.Length; i++) | ||
|  |         { | ||
|  |             for (int j = i + 1; j < resultMaterials.Length; j++) | ||
|  |             { | ||
|  |                 if (resultMaterials[i].combinedMaterial == resultMaterials[j].combinedMaterial) | ||
|  |                 { | ||
|  |                     Debug.LogError(String.Format("Source To Combined Mapping: Submesh {0} and Submesh {1} use the same combined material. These should be different", i, j)); | ||
|  |                     return false; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             MB_MultiMaterial mm = resultMaterials[i]; | ||
|  |             if (mm.combinedMaterial == null) | ||
|  |             { | ||
|  |                 Debug.LogError("Combined Material is null please create and assign a result material."); | ||
|  |                 return false; | ||
|  |             } | ||
|  |             Shader targShader = mm.combinedMaterial.shader; | ||
|  |             for (int j = 0; j < mm.sourceMaterials.Count; j++) | ||
|  |             { | ||
|  |                 if (mm.sourceMaterials[j] == null) | ||
|  |                 { | ||
|  |                     Debug.LogError("There are null entries in the list of Source Materials"); | ||
|  |                     return false; | ||
|  |                 } | ||
|  |                 if (targShader != mm.sourceMaterials[j].shader) | ||
|  |                 { | ||
|  |                     Debug.LogWarning("Source material " + mm.sourceMaterials[j] + " does not use shader " + targShader + " it may not have the required textures. If not empty textures will be generated."); | ||
|  |                 } | ||
|  |                 if (allMatsInMapping.Contains(mm.sourceMaterials[j])) | ||
|  |                 { | ||
|  |                     Debug.LogError("A Material " + mm.sourceMaterials[j] + " appears more than once in the list of source materials in the source material to combined mapping. Each source material must be unique."); | ||
|  |                     return false; | ||
|  |                 } | ||
|  |                 allMatsInMapping.Add(mm.sourceMaterials[j]); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         if (allMatsOnObjs.IsProperSubsetOf(allMatsInMapping)) | ||
|  |         { | ||
|  |             allMatsInMapping.ExceptWith(allMatsOnObjs); | ||
|  |             Debug.LogWarning("There are materials in the mapping that are not used on your source objects: " + PrintSet(allMatsInMapping)); | ||
|  |         } | ||
|  |         if (resultMaterials != null && resultMaterials.Length > 0 && allMatsInMapping.IsProperSubsetOf(allMatsOnObjs)) | ||
|  |         { | ||
|  |             allMatsOnObjs.ExceptWith(allMatsInMapping); | ||
|  |             Debug.LogError("There are materials on the objects to combine that are not in the mapping: " + PrintSet(allMatsOnObjs)); | ||
|  |             return false; | ||
|  |         } | ||
|  |         return true; | ||
|  |     } | ||
|  | } | ||
|  | 
 |