using System.Collections.Generic; using System.Linq; using UnityEngine.Scripting.APIUpdating; using UnityEngine.U2D.Common; namespace UnityEngine.U2D.Animation { /// /// Component that holds a Sprite Library Asset. The component is used by SpriteResolver Component to query for Sprite based on Category and Index. /// [DisallowMultipleComponent] [AddComponentMenu("2D Animation/Sprite Library")] [IconAttribute(IconUtility.IconPath + "Animation.SpriteLibrary.png")] [MovedFrom("UnityEngine.Experimental.U2D.Animation")] [HelpURL("https://docs.unity3d.com/Packages/com.unity.2d.animation@9.0/manual/SL-component.html")] public class SpriteLibrary : MonoBehaviour, IPreviewable { struct CategoryEntrySprite { public string category; public string entry; public Sprite sprite; } [SerializeField] List m_Library = new List(); [SerializeField] SpriteLibraryAsset m_SpriteLibraryAsset; // Cache for combining data in sprite library asset and main library Dictionary m_CategoryEntryHashCache = null; Dictionary> m_CategoryEntryCache = null; int m_PreviousSpriteLibraryAsset; long m_PreviousModificationHash; /// Get or Set the current SpriteLibraryAsset to use. public SpriteLibraryAsset spriteLibraryAsset { set { if (m_SpriteLibraryAsset != value) { m_SpriteLibraryAsset = value; CacheOverrides(); RefreshSpriteResolvers(); } } get { return m_SpriteLibraryAsset; } } void OnEnable() { CacheOverrides(); } /// /// Empty method. Implemented for the IPreviewable interface. /// public void OnPreviewUpdate() { } /// /// Return the Sprite that is registered for the given Category and Label for the SpriteLibrary. /// /// Category name. /// Label name. /// Sprite associated to the name and index. public Sprite GetSprite(string category, string label) { return GetSprite(GetHashForCategoryAndEntry(category, label)); } Sprite GetSprite(int hash) { if (m_CategoryEntryHashCache.ContainsKey(hash)) return m_CategoryEntryHashCache[hash].sprite; return null; } void UpdateCacheOverridesIfNeeded() { if(m_CategoryEntryCache == null || m_PreviousSpriteLibraryAsset != m_SpriteLibraryAsset?.GetInstanceID() || m_PreviousModificationHash != m_SpriteLibraryAsset?.modificationHash) CacheOverrides(); } internal bool GetCategoryAndEntryNameFromHash(int hash, out string category, out string entry) { UpdateCacheOverridesIfNeeded(); if (m_CategoryEntryHashCache.ContainsKey(hash)) { category = m_CategoryEntryHashCache[hash].category; entry = m_CategoryEntryHashCache[hash].entry; return true; } category = null; entry = null; return false; } internal static int GetHashForCategoryAndEntry(string category, string entry) { return SpriteLibraryUtility.GetStringHash($"{category}_{entry}"); } internal Sprite GetSpriteFromCategoryAndEntryHash(int hash, out bool validEntry) { UpdateCacheOverridesIfNeeded(); if (m_CategoryEntryHashCache.ContainsKey(hash)) { validEntry = true; return m_CategoryEntryHashCache[hash].sprite; } validEntry = false; return null; } List GetEntries(string category, bool addIfNotExist) { var index = m_Library.FindIndex(x => x.name == category); if (index < 0) { if (!addIfNotExist) return null; m_Library.Add(new SpriteLibCategory() { name = category, categoryList = new List() }); index = m_Library.Count - 1; } return m_Library[index].categoryList; } static SpriteCategoryEntry GetEntry(List entries, string entry, bool addIfNotExist) { var index = entries.FindIndex(x => x.name == entry); if (index < 0) { if (!addIfNotExist) return null; entries.Add(new SpriteCategoryEntry() { name = entry, }); index = entries.Count - 1; } return entries[index]; } /// /// Add or replace an override when querying for the given Category and Label from a SpriteLibraryAsset. /// /// Sprite Library Asset to query. /// Category name from the Sprite Library Asset to add override. /// Label name to add override. public void AddOverride(SpriteLibraryAsset spriteLib, string category, string label) { var sprite = spriteLib.GetSprite(category, label); var entries = GetEntries(category, true); var entry = GetEntry(entries, label, true); entry.sprite = sprite; CacheOverrides(); } /// /// Add or replace an override when querying for the given Category. All the categories in the Category will be added. /// /// Sprite Library Asset to query. /// Category name from the Sprite Library Asset to add override. public void AddOverride(SpriteLibraryAsset spriteLib, string category) { var categoryHash = SpriteLibraryUtility.GetStringHash(category); var cat = spriteLib.categories.FirstOrDefault(x => x.hash == categoryHash); if (cat != null) { var entries = GetEntries(category, true); for (var i = 0; i < cat.categoryList.Count; ++i) { var ent = cat.categoryList[i]; GetEntry(entries, ent.name, true).sprite = ent.sprite; } CacheOverrides(); } } /// /// Add or replace an override when querying for the given Category and Label. /// /// Sprite to override to. /// Category name to override. /// Label name to override. public void AddOverride(Sprite sprite, string category, string label) { GetEntry(GetEntries(category, true), label, true).sprite = sprite; CacheOverrides(); RefreshSpriteResolvers(); } /// /// Remove all Sprite Library override for a given category. /// /// Category overrides to remove. public void RemoveOverride(string category) { var index = m_Library.FindIndex(x => x.name == category); if (index >= 0) { m_Library.RemoveAt(index); CacheOverrides(); RefreshSpriteResolvers(); } } /// /// Remove Sprite Library override for a given category and label. /// /// Category to remove. /// Label to remove. public void RemoveOverride(string category, string label) { var entries = GetEntries(category, false); if (entries != null) { var index = entries.FindIndex(x => x.name == label); if (index >= 0) { entries.RemoveAt(index); CacheOverrides(); RefreshSpriteResolvers(); } } } /// /// Method to check if a Category and Label pair has an override. /// /// Category name. /// Label name. /// True if override exist, false otherwise. public bool HasOverride(string category, string label) { var catOverride = GetEntries(category, false); if (catOverride != null) return GetEntry(catOverride, label, false) != null; return false; } /// /// Request SpriteResolver components that are in the same hierarchy to refresh. /// public void RefreshSpriteResolvers() { var spriteResolvers = GetComponentsInChildren(); foreach (var sr in spriteResolvers) { sr.ResolveSpriteToSpriteRenderer(); #if UNITY_EDITOR sr.spriteLibChanged = true; #endif } } internal IEnumerable categoryNames { get { UpdateCacheOverridesIfNeeded(); return m_CategoryEntryCache.Keys; } } internal IEnumerable GetEntryNames(string category) { UpdateCacheOverridesIfNeeded(); if (m_CategoryEntryCache.ContainsKey(category)) return m_CategoryEntryCache[category]; return null; } internal void CacheOverrides() { m_PreviousSpriteLibraryAsset = 0; m_PreviousModificationHash = 0; m_CategoryEntryHashCache = new Dictionary(); m_CategoryEntryCache = new Dictionary>(); if (m_SpriteLibraryAsset) { m_PreviousSpriteLibraryAsset = m_SpriteLibraryAsset.GetInstanceID(); m_PreviousModificationHash = m_SpriteLibraryAsset.modificationHash; foreach (var category in m_SpriteLibraryAsset.categories) { var catName = category.name; m_CategoryEntryCache.Add(catName, new HashSet()); var cacheEntryName = m_CategoryEntryCache[catName]; foreach (var entry in category.categoryList) { m_CategoryEntryHashCache.Add(GetHashForCategoryAndEntry(catName, entry.name), new CategoryEntrySprite() { category = catName, entry = entry.name, sprite = entry.sprite }); cacheEntryName.Add(entry.name); } } } foreach (var category in m_Library) { var catName = category.name; if(!m_CategoryEntryCache.ContainsKey(catName)) m_CategoryEntryCache.Add(catName, new HashSet()); var cacheEntryName = m_CategoryEntryCache[catName]; foreach (var ent in category.categoryList) { if (!cacheEntryName.Contains(ent.name)) cacheEntryName.Add(ent.name); var hash = GetHashForCategoryAndEntry(catName, ent.name); if (!m_CategoryEntryHashCache.ContainsKey(hash)) { m_CategoryEntryHashCache.Add(hash, new CategoryEntrySprite() { category = catName, entry = ent.name, sprite = ent.sprite }); } else { var e = m_CategoryEntryHashCache[hash]; e.sprite = ent.sprite; m_CategoryEntryHashCache[hash] = e; } } } } } }