using System.Collections.Generic; using System.Drawing.Imaging; using System.IO; using System.Linq; using System.Text; using AssetStudio; using TGASharpLib; namespace AssetStudioGUI { internal static class Exporter { public static bool ExportTexture2D(AssetItem item, string exportPathName) { var m_Texture2D = (Texture2D)item.Asset; if (Properties.Settings.Default.convertTexture) { var bitmap = m_Texture2D.ConvertToBitmap(true); if (bitmap == null) return false; ImageFormat format = null; var ext = Properties.Settings.Default.convertType; bool tga = false; switch (ext) { case "BMP": format = ImageFormat.Bmp; break; case "PNG": format = ImageFormat.Png; break; case "JPEG": format = ImageFormat.Jpeg; break; case "TGA": tga = true; break; } var exportFullName = exportPathName + item.Text + "." + ext.ToLower(); if (ExportFileExists(exportFullName)) return false; if (tga) { var file = new TGA(bitmap); file.Save(exportFullName); } else bitmap.Save(exportFullName, format); bitmap.Dispose(); return true; } else { var exportFullName = exportPathName + item.Text + ".tex"; if (ExportFileExists(exportFullName)) return false; File.WriteAllBytes(exportFullName, m_Texture2D.image_data.GetData()); return true; } } public static bool ExportAudioClip(AssetItem item, string exportPath) { var m_AudioClip = (AudioClip)item.Asset; var m_AudioData = m_AudioClip.m_AudioData.GetData(); if (m_AudioData == null || m_AudioData.Length == 0) return false; var converter = new AudioClipConverter(m_AudioClip); if (Properties.Settings.Default.convertAudio && converter.IsSupport) { var exportFullName = exportPath + item.Text + ".wav"; if (ExportFileExists(exportFullName)) return false; var buffer = converter.ConvertToWav(); if (buffer == null) return false; File.WriteAllBytes(exportFullName, buffer); } else { var exportFullName = exportPath + item.Text + converter.GetExtensionName(); if (ExportFileExists(exportFullName)) return false; File.WriteAllBytes(exportFullName, m_AudioData); } return true; } public static bool ExportShader(AssetItem item, string exportPath) { var exportFullName = exportPath + item.Text + ".shader"; if (ExportFileExists(exportFullName)) return false; var m_Shader = (Shader)item.Asset; if (m_Shader.compressedBlob != null) //5.5 and up { var strs = ShaderConverter.ConvertMultiple(m_Shader); for (int i = 0; i < strs.Length; i++) { var platformName = ShaderConverter.GetPlatformString(m_Shader.platforms[i]); File.WriteAllText($"{exportPath}{item.Text}_{platformName}.shader", strs[i]); } } else { var str = ShaderConverter.Convert(m_Shader); File.WriteAllText(exportFullName, str); } return true; } public static bool ExportTextAsset(AssetItem item, string exportPath) { var m_TextAsset = (TextAsset)(item.Asset); var extension = ".txt"; if (Properties.Settings.Default.restoreExtensionName) { if (!string.IsNullOrEmpty(item.Container)) { extension = Path.GetExtension(item.Container); } } var exportFullName = exportPath + item.Text + extension; if (ExportFileExists(exportFullName)) return false; File.WriteAllBytes(exportFullName, m_TextAsset.m_Script); return true; } public static bool ExportMonoBehaviour(AssetItem item, string exportPath) { var exportFullName = exportPath + item.Text + ".txt"; if (ExportFileExists(exportFullName)) return false; var m_MonoBehaviour = (MonoBehaviour)item.Asset; var str = m_MonoBehaviour.Dump() ?? Studio.GetScriptString(item.Asset.reader); File.WriteAllText(exportFullName, str); return true; } public static bool ExportFont(AssetItem item, string exportPath) { var m_Font = (Font)item.Asset; if (m_Font.m_FontData != null) { var extension = ".ttf"; if (m_Font.m_FontData[0] == 79 && m_Font.m_FontData[1] == 84 && m_Font.m_FontData[2] == 84 && m_Font.m_FontData[3] == 79) { extension = ".otf"; } var exportFullName = exportPath + item.Text + extension; if (ExportFileExists(exportFullName)) return false; File.WriteAllBytes(exportFullName, m_Font.m_FontData); return true; } return false; } public static bool ExportMesh(AssetItem item, string exportPath) { var m_Mesh = (Mesh)item.Asset; if (m_Mesh.m_VertexCount <= 0) return false; var exportFullName = exportPath + item.Text + ".obj"; if (ExportFileExists(exportFullName)) return false; var sb = new StringBuilder(); sb.AppendLine("g " + m_Mesh.m_Name); #region Vertices if (m_Mesh.m_Vertices == null || m_Mesh.m_Vertices.Length == 0) { return false; } int c = 3; if (m_Mesh.m_Vertices.Length == m_Mesh.m_VertexCount * 4) { c = 4; } for (int v = 0; v < m_Mesh.m_VertexCount; v++) { sb.AppendFormat("v {0} {1} {2}\r\n", -m_Mesh.m_Vertices[v * c], m_Mesh.m_Vertices[v * c + 1], m_Mesh.m_Vertices[v * c + 2]); } #endregion #region UV if (m_Mesh.m_UV0?.Length > 0) { if (m_Mesh.m_UV0.Length == m_Mesh.m_VertexCount * 2) { c = 2; } else if (m_Mesh.m_UV0.Length == m_Mesh.m_VertexCount * 3) { c = 3; } for (int v = 0; v < m_Mesh.m_VertexCount; v++) { sb.AppendFormat("vt {0} {1}\r\n", m_Mesh.m_UV0[v * c], m_Mesh.m_UV0[v * c + 1]); } } #endregion #region Normals if (m_Mesh.m_Normals?.Length > 0) { if (m_Mesh.m_Normals.Length == m_Mesh.m_VertexCount * 3) { c = 3; } else if (m_Mesh.m_Normals.Length == m_Mesh.m_VertexCount * 4) { c = 4; } for (int v = 0; v < m_Mesh.m_VertexCount; v++) { sb.AppendFormat("vn {0} {1} {2}\r\n", -m_Mesh.m_Normals[v * c], m_Mesh.m_Normals[v * c + 1], m_Mesh.m_Normals[v * c + 2]); } } #endregion #region Face int sum = 0; for (var i = 0; i < m_Mesh.m_SubMeshes.Length; i++) { sb.AppendLine($"g {m_Mesh.m_Name}_{i}"); int indexCount = (int)m_Mesh.m_SubMeshes[i].indexCount; var end = sum + indexCount / 3; for (int f = sum; f < end; f++) { sb.AppendFormat("f {0}/{0}/{0} {1}/{1}/{1} {2}/{2}/{2}\r\n", m_Mesh.m_Indices[f * 3 + 2] + 1, m_Mesh.m_Indices[f * 3 + 1] + 1, m_Mesh.m_Indices[f * 3] + 1); } sum = end; } #endregion sb.Replace("NaN", "0"); File.WriteAllText(exportFullName, sb.ToString()); return true; } public static bool ExportVideoClip(AssetItem item, string exportPath) { var m_VideoClip = (VideoClip)item.Asset; var m_VideoData = m_VideoClip.m_VideoData.GetData(); if (m_VideoData != null && m_VideoData.Length != 0) { var exportFullName = exportPath + item.Text + Path.GetExtension(m_VideoClip.m_OriginalPath); if (ExportFileExists(exportFullName)) return false; File.WriteAllBytes(exportFullName, m_VideoData); return true; } return false; } public static bool ExportMovieTexture(AssetItem item, string exportPath) { var m_MovieTexture = (MovieTexture)item.Asset; var exportFullName = exportPath + item.Text + ".ogv"; if (ExportFileExists(exportFullName)) return false; File.WriteAllBytes(exportFullName, m_MovieTexture.m_MovieData); return true; } public static bool ExportSprite(AssetItem item, string exportPath) { ImageFormat format = null; var type = Properties.Settings.Default.convertType; bool tga = false; switch (type) { case "BMP": format = ImageFormat.Bmp; break; case "PNG": format = ImageFormat.Png; break; case "JPEG": format = ImageFormat.Jpeg; break; case "TGA": tga = true; break; } var exportFullName = exportPath + item.Text + "." + type.ToLower(); if (ExportFileExists(exportFullName)) return false; var bitmap = SpriteHelper.GetImageFromSprite((Sprite)item.Asset); if (bitmap != null) { if (tga) { var file = new TGA(bitmap); file.Save(exportFullName); } else bitmap.Save(exportFullName, format); bitmap.Dispose(); return true; } return false; } public static bool ExportRawFile(AssetItem item, string exportPath) { var exportFullName = exportPath + item.Text + ".dat"; if (ExportFileExists(exportFullName)) return false; File.WriteAllBytes(exportFullName, item.Asset.GetRawData()); return true; } private static bool ExportFileExists(string filename) { if (File.Exists(filename)) { return true; } Directory.CreateDirectory(Path.GetDirectoryName(filename)); return false; } public static bool ExportAnimator(AssetItem item, string exportPath, List animationList = null) { var m_Animator = (Animator)item.Asset; var convert = animationList != null ? new ModelConverter(m_Animator, animationList.Select(x => (AnimationClip)x.Asset).ToArray()) : new ModelConverter(m_Animator); exportPath = $"{exportPath}{item.Text}\\{item.Text}.fbx"; ExportFbx(convert, exportPath); return true; } public static void ExportGameObject(GameObject gameObject, string exportPath, List animationList = null) { var convert = animationList != null ? new ModelConverter(gameObject, animationList.Select(x => (AnimationClip)x.Asset).ToArray()) : new ModelConverter(gameObject); exportPath = exportPath + Studio.FixFileName(gameObject.m_Name) + ".fbx"; ExportFbx(convert, exportPath); } public static void ExportGameObjectMerge(List gameObject, string exportPath, List animationList = null) { var rootName = Path.GetFileNameWithoutExtension(exportPath); var convert = animationList != null ? new ModelConverter(rootName, gameObject, animationList.Select(x => (AnimationClip)x.Asset).ToArray()) : new ModelConverter(rootName, gameObject); ExportFbx(convert, exportPath); } private static void ExportFbx(IImported convert, string exportPath) { var eulerFilter = Properties.Settings.Default.eulerFilter; var filterPrecision = (float)Properties.Settings.Default.filterPrecision; var exportAllNodes = Properties.Settings.Default.exportAllNodes; var exportSkins = Properties.Settings.Default.exportSkins; var exportAnimations = Properties.Settings.Default.exportAnimations; var exportBlendShape = Properties.Settings.Default.exportBlendShape; var castToBone = Properties.Settings.Default.castToBone; var boneSize = (int)Properties.Settings.Default.boneSize; var scaleFactor = (float)Properties.Settings.Default.scaleFactor; var fbxVersion = Properties.Settings.Default.fbxVersion; var fbxFormat = Properties.Settings.Default.fbxFormat; ModelExporter.ExportFbx(exportPath, convert, eulerFilter, filterPrecision, exportAllNodes, exportSkins, exportAnimations, exportBlendShape, castToBone, boneSize, scaleFactor, fbxVersion, fbxFormat == 1); } public static bool ExportDumpFile(AssetItem item, string exportPath) { var exportFullName = exportPath + item.Text + ".txt"; if (ExportFileExists(exportFullName)) return false; var str = item.Asset.Dump(); if (str != null) { File.WriteAllText(exportFullName, str); return true; } return false; } public static bool ExportConvertFile(AssetItem item, string exportPath) { switch (item.Type) { case ClassIDType.Texture2D: return ExportTexture2D(item, exportPath); case ClassIDType.AudioClip: return ExportAudioClip(item, exportPath); case ClassIDType.Shader: return ExportShader(item, exportPath); case ClassIDType.TextAsset: return ExportTextAsset(item, exportPath); case ClassIDType.MonoBehaviour: return ExportMonoBehaviour(item, exportPath); case ClassIDType.Font: return ExportFont(item, exportPath); case ClassIDType.Mesh: return ExportMesh(item, exportPath); case ClassIDType.VideoClip: return ExportVideoClip(item, exportPath); case ClassIDType.MovieTexture: return ExportMovieTexture(item, exportPath); case ClassIDType.Sprite: return ExportSprite(item, exportPath); case ClassIDType.Animator: return ExportAnimator(item, exportPath); case ClassIDType.AnimationClip: return false; default: return ExportRawFile(item, exportPath); } } } }