using System; using System.Collections.Generic; using System.Linq; using UnityEngine; namespace UnityEditor.U2D.Animation { internal class Cache : BaseObject, ICacheUndo { public static T Create() where T : Cache { var cache = CreateInstance(); cache.hideFlags = HideFlags.DontSave; cache.name = cache.GetType().ToString(); return cache; } public static void Destroy(Cache cache) { cache.Destroy(); DestroyImmediate(cache); } [SerializeField] private List m_CacheObjects = new List(); [SerializeField] private List m_RemovedCacheObjects = new List(); private string m_UndoOperationName = null; private IUndo m_DefaultUndo = new UnityEngineUndo(); private IUndo m_UndoOverride = null; protected IUndo undo { get { if (undoOverride != null) return undoOverride; return m_DefaultUndo; } } public IUndo undoOverride { get { return m_UndoOverride; } set { m_UndoOverride = value; } } public bool isUndoOperationSet { get { return string.IsNullOrEmpty(m_UndoOperationName) == false; } } public void IncrementCurrentGroup() { undo.IncrementCurrentGroup(); } public virtual void BeginUndoOperation(string operationName) { if (isUndoOperationSet == false) { Debug.Assert(!m_CacheObjects.Contains(null)); m_UndoOperationName = operationName; undo.RegisterCompleteObjectUndo(this, m_UndoOperationName); undo.RegisterCompleteObjectUndo(m_CacheObjects.ToArray(), m_UndoOperationName); undo.RegisterCompleteObjectUndo(m_RemovedCacheObjects.ToArray(), m_UndoOperationName); } } public void EndUndoOperation() { m_UndoOperationName = null; } public bool IsRemoved(CacheObject cacheObject) { return m_RemovedCacheObjects.Contains(cacheObject); } public T CreateCache() where T : CacheObject { var cacheObject = FindRemovedCacheObject(); if (cacheObject != null) { m_RemovedCacheObjects.Remove(cacheObject); cacheObject.OnEnable(); } else { cacheObject = CacheObject.Create(this); } m_CacheObjects.Add(cacheObject); cacheObject.OnCreate(); return cacheObject; } private T FindRemovedCacheObject() where T : CacheObject { return m_RemovedCacheObjects.FirstOrDefault((o) => o.GetType().Equals(typeof(T))) as T; } public void Destroy(CacheObject cacheObject) { Debug.Assert(cacheObject != null); Debug.Assert(cacheObject.owner == this); Debug.Assert(m_CacheObjects.Contains(cacheObject)); m_CacheObjects.Remove(cacheObject); m_RemovedCacheObjects.Add(cacheObject); cacheObject.OnDestroy(); } public void Destroy() { Debug.Assert(!m_CacheObjects.Contains(null)); EndUndoOperation(); undo.ClearUndo(this); var cacheObjects = m_CacheObjects.ToArray(); foreach (var cacheObject in cacheObjects) DestroyImmediate(cacheObject); cacheObjects = m_RemovedCacheObjects.ToArray(); foreach (var cacheObject in cacheObjects) DestroyImmediate(cacheObject); m_CacheObjects.Clear(); m_RemovedCacheObjects.Clear(); } } }