using System; using System.Collections.Generic; using Unity.Profiling; using UnityEngine; using UnityEngine.Perception.Randomization.Samplers; namespace UnityEngine.Perception.Randomization.Randomizers.Utilities { /// /// Facilitates object pooling for a pre-specified collection of prefabs with the caveat that objects can be fetched /// from the cache but not returned. Every frame, the cache needs to be reset, which will return all objects to the pool /// public class GameObjectOneWayCache { static ProfilerMarker s_ResetAllObjectsMarker = new ProfilerMarker("ResetAllObjects"); GameObject[] m_Prefabs; UniformSampler m_Sampler = new UniformSampler(); Transform m_CacheParent; Dictionary m_InstanceIdToIndex; List[] m_InstantiatedObjects; int[] m_NumObjectsActive; int NumObjectsInCache { get; set; } /// /// The number of active cache objects in the scene /// public int NumObjectsActive { get; private set; } /// /// Creates a new GameObjectOneWayCache /// /// The parent object all cached instances will be parented under /// The prefabs to cache public GameObjectOneWayCache(Transform parent, GameObject[] prefabs) { m_Prefabs = prefabs; m_CacheParent = parent; m_InstanceIdToIndex = new Dictionary(); m_InstantiatedObjects = new List[prefabs.Length]; m_NumObjectsActive = new int[prefabs.Length]; var index = 0; foreach (var prefab in prefabs) { if (!IsPrefab(prefab)) { prefab.transform.parent = parent; prefab.SetActive(false); } var instanceId = prefab.GetInstanceID(); m_InstanceIdToIndex.Add(instanceId, index); m_InstantiatedObjects[index] = new List(); m_NumObjectsActive[index] = 0; ++index; } } /// /// Retrieves an existing instance of the given prefab from the cache if available. /// Otherwise, instantiate a new instance of the given prefab. /// /// /// /// public GameObject GetOrInstantiate(GameObject prefab) { if (!m_InstanceIdToIndex.TryGetValue(prefab.GetInstanceID(), out var index)) throw new ArgumentException($"Prefab {prefab.name} (ID: {prefab.GetInstanceID()}) is not in cache."); ++NumObjectsActive; if (m_NumObjectsActive[index] < m_InstantiatedObjects[index].Count) { var nextInCache = m_InstantiatedObjects[index][m_NumObjectsActive[index]]; ++m_NumObjectsActive[index]; foreach (var tag in nextInCache.randomizerTags) tag.Register(); return nextInCache.instance; } ++NumObjectsInCache; var newObject = Object.Instantiate(prefab, m_CacheParent); newObject.SetActive(true); ++m_NumObjectsActive[index]; m_InstantiatedObjects[index].Add(new CachedObjectData(newObject)); return newObject; } /// /// Retrieves an existing instance of the given prefab from the cache if available. /// Otherwise, instantiate a new instance of the given prefab. /// /// The index of the prefab to instantiate /// /// public GameObject GetOrInstantiate(int index) { var prefab = m_Prefabs[index]; return GetOrInstantiate(prefab); } /// /// Retrieves an existing instance of a random prefab from the cache if available. /// Otherwise, instantiate a new instance of the random prefab. /// /// public GameObject GetOrInstantiateRandomPrefab() { return GetOrInstantiate(m_Prefabs[(int)(m_Sampler.Sample() * m_Prefabs.Length)]); } /// /// Return all active cache objects back to an inactive state /// public void ResetAllObjects() { using (s_ResetAllObjectsMarker.Auto()) { NumObjectsActive = 0; for (var i = 0; i < m_InstantiatedObjects.Length; ++i) { m_NumObjectsActive[i] = 0; foreach (var cachedObjectData in m_InstantiatedObjects[i]) { // Position outside the frame cachedObjectData.instance.transform.localPosition = new Vector3(10000, 0, 0); foreach (var tag in cachedObjectData.randomizerTags) tag.Unregister(); } } } } static bool IsPrefab(GameObject obj) { return obj.scene.rootCount == 0; } struct CachedObjectData { public GameObject instance; public RandomizerTag[] randomizerTags; public CachedObjectData(GameObject instance) { this.instance = instance; randomizerTags = instance.GetComponents(); } } } }