using System; using System.Collections.Generic; using System.Linq; using Unity.Collections; using UnityEngine; using UnityEditor.AssetImporters; using UnityEditor.U2D.Aseprite.Common; using UnityEditor.U2D.Sprites; using UnityEngine.Serialization; namespace UnityEditor.U2D.Aseprite { /// /// ScriptedImporter to import Aseprite files /// // Version using unity release + 5 digit padding for future upgrade. Eg 2021.2 -> 21200000 [ScriptedImporter(21300002, new string[] {"aseprite", "ase"}, AllowCaching = true)] [HelpURL("https://docs.unity3d.com/Packages/com.unity.2d.aseprite@latest")] public partial class AsepriteImporter : ScriptedImporter, ISpriteEditorDataProvider { [SerializeField] TextureImporterSettings m_TextureImporterSettings = new TextureImporterSettings() { mipmapEnabled = false, mipmapFilter = TextureImporterMipFilter.BoxFilter, sRGBTexture = true, borderMipmap = false, mipMapsPreserveCoverage = false, alphaTestReferenceValue = 0.5f, readable = false, #if ENABLE_TEXTURE_STREAMING streamingMipmaps = false, streamingMipmapsPriority = 0, #endif fadeOut = false, mipmapFadeDistanceStart = 1, mipmapFadeDistanceEnd = 3, convertToNormalMap = false, heightmapScale = 0.25F, normalMapFilter = 0, generateCubemap = TextureImporterGenerateCubemap.AutoCubemap, cubemapConvolution = 0, seamlessCubemap = false, npotScale = TextureImporterNPOTScale.ToNearest, spriteMode = (int) SpriteImportMode.Multiple, spriteExtrude = 1, spriteMeshType = SpriteMeshType.Tight, spriteAlignment = (int) SpriteAlignment.Center, spritePivot = new Vector2(0.5f, 0.5f), spritePixelsPerUnit = 100.0f, spriteBorder = new Vector4(0.0f, 0.0f, 0.0f, 0.0f), alphaSource = TextureImporterAlphaSource.FromInput, alphaIsTransparency = true, spriteTessellationDetail = -1.0f, textureType = TextureImporterType.Sprite, textureShape = TextureImporterShape.Texture2D, filterMode = FilterMode.Point, aniso = 1, mipmapBias = 0.0f, wrapModeU = TextureWrapMode.Clamp, wrapModeV = TextureWrapMode.Clamp, wrapModeW = TextureWrapMode.Clamp }; [SerializeField] AsepriteImporterSettings m_PreviousAsepriteImporterSettings; [SerializeField] AsepriteImporterSettings m_AsepriteImporterSettings = new AsepriteImporterSettings() { fileImportMode = FileImportModes.AnimatedSprite, importHiddenLayers = false, layerImportMode = LayerImportModes.MergeFrame, defaultPivotAlignment = SpriteAlignment.BottomCenter, defaultPivotSpace = PivotSpaces.Canvas, customPivotPosition = new Vector2(0.5f, 0.5f), mosaicPadding = 4, spritePadding = 0, generateAnimationClips = true, generateModelPrefab = true, addSortingGroup = true, addShadowCasters = false }; // Use for inspector to check if the file node is checked [SerializeField] #pragma warning disable 169, 414 bool m_ImportFileNodeState = true; // Used by platform settings to mark it dirty so that it will trigger a reimport [SerializeField] #pragma warning disable 169, 414 long m_PlatformSettingsDirtyTick; [SerializeField] string m_TextureAssetName = null; [SerializeField] List m_SingleSpriteImportData = new List(1) { new SpriteMetaData() }; [FormerlySerializedAs("m_MultiSpriteImportData")] [SerializeField] List m_AnimatedSpriteImportData = new List(); [SerializeField] List m_SpriteSheetImportData = new List(); [SerializeField] List m_AsepriteLayers = new List(); [SerializeField] List m_PlatformSettings = new List(); [SerializeField] bool m_GeneratePhysicsShape = false; [SerializeField] SecondarySpriteTexture[] m_SecondarySpriteTextures; [SerializeField] string m_SpritePackingTag = ""; SpriteImportMode spriteImportModeToUse => m_TextureImporterSettings.textureType != TextureImporterType.Sprite ? SpriteImportMode.None : (SpriteImportMode)m_TextureImporterSettings.spriteMode; AsepriteImportData m_ImportData; AsepriteFile m_AsepriteFile; List m_Tags = new List(); List m_Frames = new List(); [SerializeField] Vector2Int m_CanvasSize; GameObject m_RootGameObject; readonly Dictionary m_LayerIdToGameObject = new Dictionary(); AsepriteImportData importData { get { var returnValue = m_ImportData; if (returnValue == null) // Using LoadAllAssetsAtPath because AsepriteImportData is hidden returnValue = AssetDatabase.LoadAllAssetsAtPath(assetPath).FirstOrDefault(x => x is AsepriteImportData) as AsepriteImportData; if (returnValue == null) returnValue = ScriptableObject.CreateInstance(); m_ImportData = returnValue; return returnValue; } } internal bool isNPOT => Mathf.IsPowerOfTwo(importData.textureActualWidth) && Mathf.IsPowerOfTwo(importData.textureActualHeight); internal int textureActualWidth { get => importData.textureActualWidth; private set => importData.textureActualWidth = value; } internal int textureActualHeight { get => importData.textureActualHeight; private set => importData.textureActualHeight = value; } float definitionScale { get { var definitionScaleW = importData.importedTextureWidth / (float) textureActualWidth; var definitionScaleH = importData.importedTextureHeight / (float) textureActualHeight; return Mathf.Min(definitionScaleW, definitionScaleH); } } internal SecondarySpriteTexture[] secondaryTextures { get => m_SecondarySpriteTextures; set => m_SecondarySpriteTextures = value; } public override void OnImportAsset(AssetImportContext ctx) { if(m_ImportData == null) m_ImportData = ScriptableObject.CreateInstance(); m_ImportData.hideFlags = HideFlags.HideInHierarchy; try { m_AsepriteFile = AsepriteReader.ReadFile(ctx.assetPath); if (m_AsepriteFile == null) return; m_CanvasSize = new Vector2Int(m_AsepriteFile.width, m_AsepriteFile.height); var newLayers = RestructureImportData(in m_AsepriteFile); FilterOutLayers(ref newLayers); UpdateCellNames(ref newLayers); m_Frames = ExtractFrameData(in m_AsepriteFile); m_Tags = ExtractTagsData(in m_AsepriteFile); if (newLayers.Count == 0) return; var assetName = System.IO.Path.GetFileNameWithoutExtension(ctx.assetPath); List> cellBuffers; List cellWidth; List cellHeight; if (layerImportMode == LayerImportModes.IndividualLayers) { m_AsepriteLayers = UpdateLayers(in newLayers, in m_AsepriteLayers); ImportLayers.Import(m_AsepriteLayers, out cellBuffers, out cellWidth, out cellHeight); } else { ImportMergedLayers.Import(assetName, ref newLayers, out cellBuffers, out cellWidth, out cellHeight); // Update layers after merged, since merged import creates new layers. // The new layers should be compared and merged together with the ones existing in the meta file. m_AsepriteLayers = UpdateLayers(in newLayers, in m_AsepriteLayers); } var mosaicPad = m_AsepriteImporterSettings.mosaicPadding; var spritePad = m_AsepriteImporterSettings.fileImportMode == FileImportModes.AnimatedSprite ? m_AsepriteImporterSettings.spritePadding : 0; var requireSquarePotTexture = IsRequiringSquarePotTexture(ctx); ImagePacker.Pack(cellBuffers.ToArray(), cellWidth.ToArray(), cellHeight.ToArray(), (int)mosaicPad, spritePad, requireSquarePotTexture, out var outputImageBuffer, out var packedTextureWidth, out var packedTextureHeight, out var spriteRects, out var uvTransforms); var packOffsets = new Vector2Int[spriteRects.Length]; for (var i = 0; i < packOffsets.Length; ++i) { packOffsets[i] = new Vector2Int(uvTransforms[i].x - spriteRects[i].position.x, uvTransforms[i].y - spriteRects[i].position.y); packOffsets[i] *= -1; } var spriteImportData = UpdateSpriteImportData(in m_AsepriteLayers, spriteRects, packOffsets, uvTransforms); importData.importedTextureHeight = textureActualHeight = packedTextureHeight; importData.importedTextureWidth = textureActualWidth = packedTextureWidth; var output = TextureGeneration.Generate( ctx, outputImageBuffer, packedTextureWidth, packedTextureHeight, spriteImportData.ToArray(), in m_PlatformSettings, in m_TextureImporterSettings, m_SpritePackingTag, secondaryTextures); if (output.texture) { importData.importedTextureHeight = output.texture.height; importData.importedTextureWidth = output.texture.width; } if (output.texture != null && output.sprites != null) SetPhysicsOutline(GetDataProvider(), output.sprites, definitionScale, pixelsPerUnit, m_GeneratePhysicsShape); RegisterAssets(ctx, output); OnPostAsepriteImport?.Invoke(new ImportEventArgs(this, ctx)); outputImageBuffer.DisposeIfCreated(); foreach (var cellBuffer in cellBuffers) cellBuffer.DisposeIfCreated(); } catch (Exception e) { Debug.LogError($"Failed to import file {assetPath}. Error: {e.Message} \n{e.StackTrace}"); } finally { m_PreviousAsepriteImporterSettings = m_AsepriteImporterSettings; EditorUtility.SetDirty(this); m_AsepriteFile?.Dispose(); } } List RestructureImportData(in AsepriteFile file) { var frameData = file.frameData; var nameGenerator = new UniqueNameGenerator(); var layers = new List(); var parentTable = new Dictionary(); for (var i = 0; i < frameData.Count; ++i) { var chunks = frameData[i].chunks; for (var m = 0; m < chunks.Count; ++m) { if (chunks[m].chunkType == ChunkTypes.Layer) { var layerChunk = chunks[m] as LayerChunk; var layer = new Layer(); var childLevel = layerChunk.childLevel; parentTable[childLevel] = layer; layer.parentIndex = childLevel == 0 ? -1 : parentTable[childLevel - 1].index; layer.name = nameGenerator.GetUniqueName(layerChunk.name, layer.parentIndex); layer.layerFlags = layerChunk.flags; layer.layerType = layerChunk.layerType; layer.blendMode = layerChunk.blendMode; layer.opacity = layerChunk.opacity / 255f; layer.index = layers.Count; layer.guid = Layer.GenerateGuid(layer); layers.Add(layer); } } } for (var i = 0; i < frameData.Count; ++i) { var chunks = frameData[i].chunks; for (var m = 0; m < chunks.Count; ++m) { if (chunks[m].chunkType == ChunkTypes.Cell) { var cellChunk = chunks[m] as CellChunk; var layer = layers.Find(x => x.index == cellChunk.layerIndex); if (layer == null) { Debug.LogWarning($"Could not find the layer for one of the cells. Frame Index={i}, Chunk Index={m}."); continue; } var cellType = cellChunk.cellType; if (cellType == CellTypes.LinkedCell) { var cell = new LinkedCell(); cell.frameIndex = i; cell.linkedToFrame = cellChunk.linkedToFrame; layer.linkedCells.Add(cell); } else { var cell = new Cell(); cell.frameIndex = i; cell.updatedCellRect = false; // Flip Y. Aseprite 0,0 is at Top Left. Unity 0,0 is at Bottom Left. var cellY = (m_CanvasSize.y - cellChunk.posY) - cellChunk.height; cell.cellRect = new RectInt(cellChunk.posX, cellY, cellChunk.width, cellChunk.height); cell.opacity = cellChunk.opacity / 255f; cell.blendMode = layer.blendMode; cell.image = cellChunk.image; cell.additiveSortOrder = cellChunk.zIndex; cell.name = layer.name; cell.spriteId = GUID.Generate(); var opacity = cell.opacity * layer.opacity; if ((1f - opacity) > Mathf.Epsilon) TextureTasks.AddOpacity(ref cell.image, opacity); layer.cells.Add(cell); } } } } return layers; } void FilterOutLayers(ref List layers) { for (var i = layers.Count - 1; i >= 0; --i) { var layer = layers[i]; if (!includeHiddenLayers && !ImportUtilities.IsLayerVisible(layer.index, in layers)) { DisposeCellsInLayer(layer); layers.RemoveAt(i); continue; } var cells = layer.cells; for (var m = cells.Count - 1; m >= 0; --m) { var width = cells[m].cellRect.width; var height = cells[m].cellRect.width; if (width == 0 || height == 0) cells.RemoveAt(m); else if (cells[m].image == default || !cells[m].image.IsCreated) cells.RemoveAt(m); } } } static void DisposeCellsInLayer(Layer layer) { foreach (var cell in layer.cells) { var image = cell.image; image.DisposeIfCreated(); } } static void UpdateCellNames(ref List layers) { for (var i = 0; i < layers.Count; ++i) { var cells = layers[i].cells; for (var m = 0; m < cells.Count; ++m) { var cell = cells[m]; cell.name = ImportUtilities.GetCellName(cell.name, cell.frameIndex, cells.Count); cells[m] = cell; } } } static List UpdateLayers(in List newLayers, in List oldLayers) { if (oldLayers.Count == 0) return new List(newLayers); var finalLayers = new List(oldLayers); // Remove old layers for (var i = 0; i < oldLayers.Count; ++i) { var oldLayer = oldLayers[i]; if (newLayers.FindIndex(x => x.guid == oldLayer.guid) == -1) finalLayers.Remove(oldLayer); } // Add new layers for (var i = 0; i < newLayers.Count; ++i) { var newLayer = newLayers[i]; var layerIndex = finalLayers.FindIndex(x => x.guid == newLayer.guid); if (layerIndex == -1) finalLayers.Add(newLayer); } // Update layer data for (var i = 0; i < finalLayers.Count; ++i) { var finalLayer = finalLayers[i]; var layerIndex = newLayers.FindIndex(x => x.guid == finalLayer.guid); if (layerIndex != -1) { var oldCells = finalLayer.cells; var newCells = newLayers[layerIndex].cells; for (var m = 0; m < newCells.Count; ++m) { if (m < oldCells.Count) { var oldCell = oldCells[m]; var newCell = newCells[m]; newCell.spriteId = oldCell.spriteId; #if UNITY_2023_1_OR_NEWER newCell.updatedCellRect = newCell.cellRect != oldCell.cellRect; #else newCell.updatedCellRect = !newCell.cellRect.IsEqual(oldCell.cellRect); #endif newCells[m] = newCell; } } finalLayer.cells = new List(newCells); finalLayer.linkedCells = new List(newLayers[layerIndex].linkedCells); finalLayer.index = newLayers[layerIndex].index; finalLayer.opacity = newLayers[layerIndex].opacity; finalLayer.parentIndex = newLayers[layerIndex].parentIndex; } } return finalLayers; } bool IsRequiringSquarePotTexture(AssetImportContext ctx) { var platformSettings = TextureImporterUtilities.GetPlatformTextureSettings(ctx.selectedBuildTarget, in m_PlatformSettings); return (TextureImporterFormat.PVRTC_RGB2 <= platformSettings.format && platformSettings.format <= TextureImporterFormat.PVRTC_RGBA4); } static List ExtractFrameData(in AsepriteFile file) { var noOfFrames = file.noOfFrames; var frames = new List(noOfFrames); for (var i = 0; i < noOfFrames; ++i) { var frameData = file.frameData[i]; var eventStrings = ExtractEventStringFromCells(frameData); var frame = new Frame() { duration = frameData.frameDuration, eventStrings = eventStrings }; frames.Add(frame); } return frames; } static string[] ExtractEventStringFromCells(FrameData frameData) { var chunks = frameData.chunks; var eventStrings = new HashSet(); for (var i = 0; i < chunks.Count; ++i) { if (chunks[i].chunkType != ChunkTypes.Cell) continue; var cellChunk = (CellChunk)chunks[i]; if (cellChunk.dataChunk == null) continue; var dataText = cellChunk.dataChunk.text; if (string.IsNullOrEmpty(dataText) || !dataText.StartsWith("event:")) continue; var eventString = dataText.Remove(0, "event:".Length); eventString = eventString.Trim(' '); eventStrings.Add(eventString); } return eventStrings.ToArray(); } static List ExtractTagsData(in AsepriteFile file) { var tags = new List(); var noOfFrames = file.noOfFrames; for (var i = 0; i < noOfFrames; ++i) { var frame = file.frameData[i]; var noOfChunks = frame.chunkCount; for (var m = 0; m < noOfChunks; ++m) { var chunk = frame.chunks[m]; if (chunk.chunkType != ChunkTypes.Tags) continue; var tagChunk = chunk as TagsChunk; var noOfTags = tagChunk.noOfTags; for (var n = 0; n < noOfTags; ++n) { var data = tagChunk.tagData[n]; var tag = new Tag(); tag.name = data.name; tag.noOfRepeats = data.noOfRepeats; tag.fromFrame = data.fromFrame; // Adding one more frame as Aseprite's tags seems to always be 1 short. tag.toFrame = data.toFrame + 1; tags.Add(tag); } } } return tags; } List UpdateSpriteImportData(in List layers, RectInt[] spriteRects, Vector2Int[] packOffsets, Vector2Int[] uvTransforms) { if (m_AsepriteImporterSettings.fileImportMode == FileImportModes.SpriteSheet) { return GetSpriteImportData(); } var cellLookup = new List(); for (var i = 0; i < layers.Count; ++i) cellLookup.AddRange(layers[i].cells); var spriteImportData = GetSpriteImportData(); if (spriteImportData.Count <= 0) { var newSpriteMeta = new List(); for (var i = 0; i < spriteRects.Length; ++i) { var cell = cellLookup[i]; var spriteData = CreateNewSpriteMetaData(in cell, in spriteRects[i], packOffsets[i], in uvTransforms[i]); newSpriteMeta.Add(spriteData); } spriteImportData.Clear(); spriteImportData.AddRange(newSpriteMeta); } else { // Remove old cells for (var i = spriteImportData.Count - 1; i >= 0; --i) { var spriteData = spriteImportData[i]; if (cellLookup.FindIndex(x => x.spriteId == spriteData.spriteID) == -1) spriteImportData.Remove(spriteData); } // Add new cells for (var i = 0; i < cellLookup.Count; ++i) { var cell = cellLookup[i]; if (spriteImportData.FindIndex(x => x.spriteID == cell.spriteId) == -1) { var spriteData = CreateNewSpriteMetaData(in cell, spriteRects[i], packOffsets[i], uvTransforms[i]); spriteImportData.Add(spriteData); } } // Update with new pack data for (var i = 0; i < cellLookup.Count; ++i) { var cell = cellLookup[i]; var spriteData = spriteImportData.Find(x => x.spriteID == cell.spriteId); if (spriteData != null) { var areSettingsUpdated = !m_PreviousAsepriteImporterSettings.IsDefault() && (pivotAlignment != m_PreviousAsepriteImporterSettings.defaultPivotAlignment || pivotSpace != m_PreviousAsepriteImporterSettings.defaultPivotSpace || customPivotPosition != m_PreviousAsepriteImporterSettings.customPivotPosition || spritePadding != m_PreviousAsepriteImporterSettings.spritePadding); // Update pivot if either the importer settings are updated // or the source files rect has been changed (Only for Canvas, as rect position doesn't matter in local). if (pivotSpace == PivotSpaces.Canvas && (areSettingsUpdated || cell.updatedCellRect)) { spriteData.alignment = SpriteAlignment.Custom; var cellRect = cell.cellRect; cellRect.x += packOffsets[i].x; cellRect.y += packOffsets[i].y; cellRect.width = spriteRects[i].width; cellRect.height = spriteRects[i].height; spriteData.pivot = ImportUtilities.CalculateCellPivot(cellRect, spritePadding, m_CanvasSize, pivotAlignment, customPivotPosition); } else if (pivotSpace == PivotSpaces.Local && areSettingsUpdated) { spriteData.alignment = pivotAlignment; spriteData.pivot = customPivotPosition; } spriteData.rect = new Rect(spriteRects[i].x, spriteRects[i].y, spriteRects[i].width, spriteRects[i].height); spriteData.uvTransform = uvTransforms[i]; } } } return spriteImportData; } SpriteMetaData CreateNewSpriteMetaData(in Cell cell, in RectInt spriteRect, in Vector2Int packOffset, in Vector2Int uvTransform) { var spriteData = new SpriteMetaData(); spriteData.border = Vector4.zero; if (pivotSpace == PivotSpaces.Canvas) { spriteData.alignment = SpriteAlignment.Custom; var cellRect = cell.cellRect; cellRect.x += packOffset.x; cellRect.y += packOffset.y; cellRect.width = spriteRect.width; cellRect.height = spriteRect.height; spriteData.pivot = ImportUtilities.CalculateCellPivot(cellRect, spritePadding, m_CanvasSize, pivotAlignment, customPivotPosition); } else { spriteData.alignment = pivotAlignment; spriteData.pivot = customPivotPosition; } spriteData.rect = new Rect(spriteRect.x, spriteRect.y, spriteRect.width, spriteRect.height); spriteData.spriteID = cell.spriteId; spriteData.name = cell.name; spriteData.uvTransform = uvTransform; return spriteData; } static void SetPhysicsOutline(ISpritePhysicsOutlineDataProvider physicsOutlineDataProvider, Sprite[] sprites, float definitionScale, float pixelsPerUnit, bool generatePhysicsShape) { foreach (var sprite in sprites) { var guid = sprite.GetSpriteID(); var outline = physicsOutlineDataProvider.GetOutlines(guid); var generated = false; if ((outline == null || outline.Count == 0) && generatePhysicsShape) { InternalEditorBridge.GenerateOutlineFromSprite(sprite, 0.25f, 200, true, out var defaultOutline); outline = new List(defaultOutline.Length); for (var i = 0; i < defaultOutline.Length; ++i) { outline.Add(defaultOutline[i]); } generated = true; } if (outline != null && outline.Count > 0) { // Ensure that outlines are all valid. var validOutlineCount = 0; for (var i = 0; i < outline.Count; ++i) validOutlineCount += ( (outline[i].Length > 2) ? 1 : 0 ); var index = 0; var convertedOutline = new Vector2[validOutlineCount][]; var useScale = generated ? pixelsPerUnit * definitionScale : definitionScale; var outlineOffset = Vector2.zero; outlineOffset.x = sprite.rect.width * 0.5f; outlineOffset.y = sprite.rect.height * 0.5f; for (var i = 0; i < outline.Count; ++i) { if (outline[i].Length > 2) { convertedOutline[index] = new Vector2[outline[i].Length]; for (var j = 0; j < outline[i].Length; ++j) convertedOutline[index][j] = outline[i][j] * useScale + outlineOffset; index++; } } sprite.OverridePhysicsShape(convertedOutline); } } } void RegisterAssets(AssetImportContext ctx, TextureGenerationOutput output) { if ((output.sprites == null || output.sprites.Length == 0) && output.texture == null) { Debug.LogWarning(TextContent.noSpriteOrTextureImportWarning, this); return; } var assetNameGenerator = new UniqueNameGenerator(); if (!string.IsNullOrEmpty(output.importInspectorWarnings)) { Debug.LogWarning(output.importInspectorWarnings); } if (output.importWarnings != null && output.importWarnings.Length != 0) { foreach (var warning in output.importWarnings) Debug.LogWarning(warning); } if (output.thumbNail == null) Debug.LogWarning("Thumbnail generation fail"); if (output.texture == null) { throw new Exception("Texture import fail"); } var assetName = assetNameGenerator.GetUniqueName(System.IO.Path.GetFileNameWithoutExtension(ctx.assetPath), -1, true, this); UnityEngine.Object mainAsset = null; RegisterTextureAsset(ctx, output, assetName, ref mainAsset); RegisterSprites(ctx, output, assetNameGenerator); RegisterGameObjects(ctx, output, ref mainAsset); RegisterAnimationClip(ctx, assetName, output); RegisterAnimatorController(ctx, assetName); ctx.AddObjectToAsset("AsepriteImportData", m_ImportData); ctx.SetMainObject(mainAsset); } void RegisterTextureAsset(AssetImportContext ctx, TextureGenerationOutput output, string assetName, ref UnityEngine.Object mainAsset) { var registerTextureNameId = string.IsNullOrEmpty(m_TextureAssetName) ? "Texture" : m_TextureAssetName; output.texture.name = assetName; ctx.AddObjectToAsset(registerTextureNameId, output.texture, output.thumbNail); mainAsset = output.texture; } static void RegisterSprites(AssetImportContext ctx, TextureGenerationOutput output, UniqueNameGenerator assetNameGenerator) { if (output.sprites == null) return; foreach (var sprite in output.sprites) { var spriteGuid = sprite.GetSpriteID().ToString(); var spriteAssetName = assetNameGenerator.GetUniqueName(spriteGuid, -1, false, sprite); ctx.AddObjectToAsset(spriteAssetName, sprite); } } void RegisterGameObjects(AssetImportContext ctx, TextureGenerationOutput output, ref UnityEngine.Object mainAsset) { if (output.sprites.Length == 0) return; if (m_AsepriteImporterSettings.fileImportMode != FileImportModes.AnimatedSprite) return; PrefabGeneration.Generate( ctx, output, m_AsepriteLayers, m_LayerIdToGameObject, m_CanvasSize, m_AsepriteImporterSettings, ref mainAsset, out m_RootGameObject); } void RegisterAnimationClip(AssetImportContext ctx, string assetName, TextureGenerationOutput output) { if (output.sprites.Length == 0) return; if (m_AsepriteImporterSettings.fileImportMode != FileImportModes.AnimatedSprite) return; if (!generateAnimationClips) return; var noOfFrames = m_AsepriteFile.noOfFrames; if (noOfFrames == 1) return; var sprites = output.sprites; var clips = AnimationClipGeneration.Generate( assetName, sprites, m_AsepriteFile, m_AsepriteLayers, m_Frames, m_Tags, m_LayerIdToGameObject); for (var i = 0; i < clips.Length; ++i) ctx.AddObjectToAsset(clips[i].name, clips[i]); } void RegisterAnimatorController(AssetImportContext ctx, string assetName) { if (m_AsepriteImporterSettings.fileImportMode != FileImportModes.AnimatedSprite) return; AnimatorControllerGeneration.Generate(ctx, assetName, m_RootGameObject, generateModelPrefab); } internal void Apply() { // Do this so that asset change save dialog will not show var originalValue = EditorPrefs.GetBool("VerifySavingAssets", false); EditorPrefs.SetBool("VerifySavingAssets", false); AssetDatabase.ForceReserializeAssets(new string[] { assetPath }, ForceReserializeAssetsOptions.ReserializeMetadata); EditorPrefs.SetBool("VerifySavingAssets", originalValue); } public override bool SupportsRemappedAssetType(Type type) { if (type == typeof(AnimationClip)) return true; return base.SupportsRemappedAssetType(type); } void SetPlatformTextureSettings(TextureImporterPlatformSettings platformSettings) { var index = m_PlatformSettings.FindIndex(x => x.name == platformSettings.name); if(index < 0) m_PlatformSettings.Add(platformSettings); else m_PlatformSettings[index] = platformSettings; } void SetDirty() { EditorUtility.SetDirty(this); } List GetSpriteImportData() { if (spriteImportModeToUse == SpriteImportMode.Multiple) { switch (m_AsepriteImporterSettings.fileImportMode) { case FileImportModes.SpriteSheet: return m_SpriteSheetImportData; case FileImportModes.AnimatedSprite: default: return m_AnimatedSpriteImportData; } } return m_SingleSpriteImportData; } internal SpriteRect GetSpriteData(GUID guid) { if (spriteImportModeToUse == SpriteImportMode.Multiple) { switch (m_AsepriteImporterSettings.fileImportMode) { case FileImportModes.SpriteSheet: return m_SpriteSheetImportData.FirstOrDefault(x => x.spriteID == guid); case FileImportModes.AnimatedSprite: default: return m_AnimatedSpriteImportData.FirstOrDefault(x => x.spriteID == guid); } } return m_SingleSpriteImportData[0]; } internal TextureImporterPlatformSettings[] GetAllPlatformSettings() { return m_PlatformSettings.ToArray(); } internal void ReadTextureSettings(TextureImporterSettings dest) { m_TextureImporterSettings.CopyTo(dest); } } }