using SanAndreasUnity.Behaviours.World; using SanAndreasUnity.Utilities; using System.Collections; using System.Collections.Generic; using System.IO; using System.Linq; using UnityEditor; using UnityEngine; namespace SanAndreasUnity.Editor { public class AssetExporter : EditorWindowBase { private const string DefaultFolderName = "ExportedAssets"; private string m_selectedFolder = "Assets/" + DefaultFolderName; string ModelsPath => m_selectedFolder + "/Models"; string CollisionModelsPath => m_selectedFolder + "/CollisionModels"; string MaterialsPath => m_selectedFolder + "/Materials"; string TexturesPath => m_selectedFolder + "/Textures"; string PrefabsPath => m_selectedFolder + "/Prefabs"; private CoroutineInfo m_coroutineInfo; private bool m_exportFromSelection = false; private bool m_exportCollisionMeshes = true; [MenuItem(EditorCore.MenuName + "/" + "Asset exporter")] static void Init() { var window = GetWindow(); window.Show(); } public AssetExporter() { this.titleContent = new GUIContent("Asset exporter"); } void OnGUI() { EditorGUILayout.HelpBox( "This tool can export all currenty loaded world objects as assets and prefabs.\n" + "It will store them in a separate folder, and will only export those objects that were not already exported.", MessageType.Info, true); GUILayout.Space(30); GUILayout.BeginHorizontal(); EditorGUILayout.LabelField("Folder: " + m_selectedFolder); if (GUILayout.Button("Change")) this.ChangeFolder(); GUILayout.FlexibleSpace(); GUILayout.EndHorizontal(); m_exportCollisionMeshes = EditorGUILayout.Toggle("Export collision meshes", m_exportCollisionMeshes); GUILayout.Space(30); if (GUILayout.Button("Export from world")) this.Export(false); if (GUILayout.Button("Export from selection")) this.Export(true); } void ChangeFolder() { m_selectedFolder = EditorUtility.SaveFolderPanel( "Select folder where to export files", m_selectedFolder, ""); if (string.IsNullOrWhiteSpace(m_selectedFolder)) { return; } m_selectedFolder = FileUtil.GetProjectRelativePath(m_selectedFolder); if (string.IsNullOrWhiteSpace(m_selectedFolder)) { EditorUtility.DisplayDialog("", "Folder must be inside project.", "Ok"); } } void Export(bool fromSelection) { if (this.IsCoroutineRunning(m_coroutineInfo)) return; m_exportFromSelection = fromSelection; m_coroutineInfo = this.StartCoroutine(this.ExportCoroutine(), this.Cleanup, ex => this.Cleanup()); } void Cleanup() { EditorUtility.ClearProgressBar(); } IEnumerator ExportCoroutine() { yield return null; if (string.IsNullOrWhiteSpace(m_selectedFolder)) { EditorUtility.DisplayDialog("", "Select a folder first.", "Ok"); yield break; } if (m_exportFromSelection) { if (Selection.transforms.Length == 0) { EditorUtility.DisplayDialog("", "No object selected.", "Ok"); yield break; } } var cell = Cell.Instance; if (null == cell && !m_exportFromSelection) { EditorUtility.DisplayDialog("", $"{nameof(Cell)} script not found in scene. Make sure that you started the game with the correct scene.", "Ok"); yield break; } EditorUtility.DisplayProgressBar("", "Gathering info...", 0f); Transform[] objectsToExport = m_exportFromSelection ? Selection.transforms : cell.transform.GetFirstLevelChildren().ToArray(); int numObjectsActive = 0; for (int i = 0; i < objectsToExport.Length; i++) { var child = objectsToExport[i]; if (child.gameObject.activeInHierarchy) numObjectsActive++; } EditorUtility.ClearProgressBar(); if (!EditorUtility.DisplayDialog( "", $"There are {objectsToExport.Length} objects, with {numObjectsActive} active ones.\nProceed ?", "Ok", "Cancel")) { yield break; } if (EditorApplication.isPlaying) EditorApplication.isPaused = true; EditorUtility.DisplayProgressBar("", "Creating folders...", 0f); this.CreateFolders(); EditorUtility.DisplayProgressBar("", "Creating assets...", 0f); int numExported = 0; for (int i = 0; i < objectsToExport.Length; i++) { var child = objectsToExport[i]; if (!child.gameObject.activeInHierarchy) continue; if (EditorUtility.DisplayCancelableProgressBar("", $"Creating assets ({numExported}/{numObjectsActive})... {child.name}", numExported / (float)numObjectsActive)) yield break; this.ExportAssets(child.gameObject); numExported++; yield return null; } EditorUtility.DisplayProgressBar("", "Creating prefab...", 1f); if (!m_exportFromSelection) PrefabUtility.SaveAsPrefabAsset(cell.gameObject, $"{PrefabsPath}/{cell.gameObject.name}.prefab"); else { foreach (var obj in objectsToExport) { PrefabUtility.SaveAsPrefabAsset(obj.gameObject, $"{PrefabsPath}/{obj.gameObject.name}.prefab"); } } EditorUtility.DisplayProgressBar("", "Refreshing asset database...", 1f); AssetDatabase.Refresh(); EditorUtility.ClearProgressBar(); EditorUtility.DisplayDialog("", "Finished !", "Ok"); } void CreateFolders() { string[] folders = new string[] { "Models", "Materials", "Textures", "Prefabs", "CollisionModels", }; foreach (string folder in folders) { if (!AssetDatabase.IsValidFolder(Path.Combine(m_selectedFolder, folder))) AssetDatabase.CreateFolder(m_selectedFolder, folder); } } public void ExportAssets(GameObject go) { string assetName = go.name; var meshFilters = go.GetComponentsInChildren(); for (int i = 0; i < meshFilters.Length; i++) { MeshFilter meshFilter = meshFilters[i]; string indexPath = meshFilters.Length == 1 ? "" : "-" + i; meshFilter.sharedMesh = (Mesh)CreateAssetIfNotExists(meshFilter.sharedMesh, $"{ModelsPath}/{assetName}{indexPath}.asset"); } var meshRenderers = go.GetComponentsInChildren(); for (int i = 0; i < meshRenderers.Length; i++) { ExportMeshRenderer(go, meshRenderers[i], meshRenderers.Length == 1 ? (int?)null : i); } if (m_exportCollisionMeshes) { var meshColliders = go.GetComponentsInChildren(); for (int i = 0; i < meshColliders.Length; i++) { string indexPath = meshColliders.Length == 1 ? "" : "-" + i; meshColliders[i].sharedMesh = (Mesh)CreateAssetIfNotExists(meshColliders[i].sharedMesh, $"{CollisionModelsPath}/{assetName}{indexPath}.asset"); } } //PrefabUtility.SaveAsPrefabAsset(go, $"{PrefabsPath}/{assetName}.prefab"); } public void ExportMeshRenderer(GameObject rootGo, MeshRenderer meshRenderer, int? index) { string indexPath = index.HasValue ? "-" + index.Value : ""; string assetName = rootGo.name + indexPath; var mats = meshRenderer.sharedMaterials.ToArray(); for (int i = 0; i < mats.Length; i++) { var tex = mats[i].mainTexture; if (tex != null) mats[i].mainTexture = (Texture)CreateAssetIfNotExists(tex, $"{TexturesPath}/{assetName}-{i}.asset"); mats[i] = (Material)CreateAssetIfNotExists(mats[i], $"{MaterialsPath}/{assetName}-{i}.mat"); } meshRenderer.sharedMaterials = mats; } private static Object CreateAssetIfNotExists(Object asset, string path) { if (AssetDatabase.Contains(asset)) return asset; if (File.Exists(Path.Combine(Application.dataPath + "/../", path))) return AssetDatabase.LoadMainAssetAtPath(path); AssetDatabase.CreateAsset(asset, path); return AssetDatabase.LoadMainAssetAtPath(path); } } }