/// .ase & .aseprite file format specs: /// https://github.com/aseprite/aseprite/blob/main/docs/ase-file-specs.md using System; using System.Collections.ObjectModel; using System.IO; using UnityEngine.Assertions; namespace UnityEditor.U2D.Aseprite { /// /// Parsed representation of an Aseprite file. /// Should be disposed after use. /// public class AsepriteFile : IDisposable { /// /// File size in bytes. /// public uint fileSize { get; private set; } /// /// Number of frames in the file. /// public ushort noOfFrames { get; private set; } /// /// Canvas width in pixels. /// public ushort width { get; private set; } /// /// Canvas height in pixels. /// public ushort height { get; private set; } /// /// Color depth (bits per pixel). /// public ushort colorDepth { get; private set; } internal uint flags { get; private set; } /// /// Time per frame in milliseconds. /// public ushort animSpeed { get; private set; } /// /// Palette entry (index) which represent transparent color /// in all non-background layers (only for Indexed sprites). /// public byte alphaPaletteEntry { get; private set; } /// /// Number of colors (0 means 256 for old sprites). /// public ushort noOfColors { get; private set; } /// /// Pixel width (pixel ratio is "pixel width/pixel height"). /// If this or pixel height field is zero, pixel ratio is 1:1. /// public byte pixelWidth { get; private set; } /// /// Pixel height (pixel ratio is "pixel width/pixel height"). /// If this or pixel width field is zero, pixel ratio is 1:1. /// public byte pixelHeight { get; private set; } /// /// X position of the grid. /// public short gridPosX { get; private set; } /// /// Y position of the grid. /// public short gridPosY { get; private set; } /// /// Grid width (zero if there is no grid, grid size is 16x16 on Aseprite by default). /// public ushort gridWidth { get; private set; } /// /// Grid height (zero if there is no grid). /// public ushort gridHeight { get; private set; } /// /// Parsed data of each frame. /// public ReadOnlyCollection frameData => Array.AsReadOnly(m_FrameData); FrameData[] m_FrameData; internal void Read(BinaryReader reader) { var streamLength = reader.BaseStream.Length; Assert.IsTrue(streamLength >= 128, "File is too small to be a valid Aseprite file."); fileSize = reader.ReadUInt32(); var misc0 = reader.ReadUInt16(); noOfFrames = reader.ReadUInt16(); width = reader.ReadUInt16(); height = reader.ReadUInt16(); colorDepth = reader.ReadUInt16(); flags = reader.ReadUInt32(); animSpeed = reader.ReadUInt16(); var misc1 = reader.ReadUInt32(); var misc2 = reader.ReadUInt32(); alphaPaletteEntry = reader.ReadByte(); var miscByte0 = reader.ReadByte(); var miscByte1 = reader.ReadByte(); var miscByte2 = reader.ReadByte(); noOfColors = reader.ReadUInt16(); pixelWidth = reader.ReadByte(); pixelHeight = reader.ReadByte(); gridPosX = reader.ReadInt16(); gridPosY = reader.ReadInt16(); gridWidth = reader.ReadUInt16(); gridHeight = reader.ReadUInt16(); Assert.IsTrue(misc0 == 0xA5E0, "Unexpected file content. The file is most likely corrupt."); // Unused 84 bytes for (var i = 0; i < 84; ++i) reader.ReadByte(); m_FrameData = new FrameData[noOfFrames]; } internal void SetFrameData(int index, FrameData data) { if (index < 0 || index >= m_FrameData.Length) return; m_FrameData[index] = data; } public void Dispose() { for (var i = 0; i < m_FrameData.Length; ++i) m_FrameData[i].Dispose(); } } }