在此项目中,您能够访问使用 Visual Effect Graph 制作的示例场景和效果。
您最多选择25个主题 主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

210 行
8.0 KiB

using System;
namespace UnityEngine.VFX.Utility
{
[ExecuteAlways]
[RequireComponent(typeof(VisualEffect))]
class VFXOutputEventPrefabSpawn : VFXOutputEventAbstractHandler
{
public override bool canExecuteInEditor => true;
public uint instanceCount => m_InstanceCount;
public GameObject prefabToSpawn => m_PrefabToSpawn;
public bool parentInstances => m_ParentInstances;
#pragma warning disable 414, 649
[SerializeField, Tooltip("The maximum number of prefabs that can be active at a time")]
uint m_InstanceCount = 5;
[SerializeField, Tooltip("The prefab to enable upon event received. Prefabs are created as hidden and stored in a pool, upon enabling this behavior. Upon receiving an event a prefab from the pool is enabled and will be disabled when reaching its lifetime.")]
GameObject m_PrefabToSpawn;
[SerializeField, Tooltip("Whether to attach prefab instances to current game object. Use this setting to treat position and angle attributes as local space.")]
bool m_ParentInstances;
#pragma warning restore 414, 649
#if UNITY_EDITOR
bool m_Dirty = true;
#endif
[Tooltip("Whether to use the position attribute to set prefab position on spawn")]
public bool usePosition = true;
[Tooltip("Whether to use the angle attribute to set prefab rotation on spawn")]
public bool useAngle = true;
[Tooltip("Whether to use the scale attribute to set prefab localScale on spawn")]
public bool useScale = true;
[Tooltip("Whether to use the lifetime attribute to determine how long the prefab will be enabled")]
public bool useLifetime = true;
static readonly GameObject[] k_EmptyGameObjects = new GameObject[0];
static readonly float[] k_EmptyTimeToLive = new float[0];
GameObject[] m_Instances = k_EmptyGameObjects;
float[] m_TimesToLive = k_EmptyTimeToLive;
protected override void OnDisable()
{
base.OnDisable();
foreach (var instance in m_Instances)
instance.SetActive(false);
}
void OnDestroy()
{
DisposeInstances();
}
#if UNITY_EDITOR
void OnValidate()
{
m_Dirty = true;
}
#endif
void DisposeInstances()
{
foreach (var instance in m_Instances)
{
if (instance)
{
if (Application.isPlaying)
Destroy(instance);
else
DestroyImmediate(instance);
}
}
m_Instances = k_EmptyGameObjects;
m_TimesToLive = k_EmptyTimeToLive;
}
static readonly int k_PositionID = Shader.PropertyToID("position");
static readonly int k_AngleID = Shader.PropertyToID("angle");
static readonly int k_ScaleID = Shader.PropertyToID("scale");
static readonly int k_LifetimeID = Shader.PropertyToID("lifetime");
void UpdateHideFlag(GameObject instance)
{
instance.hideFlags = HideFlags.HideAndDontSave;
//We are using HideInHierarchy to prevent unexpected deletion in edit mode.
//instance.hideFlags = HideFlags.DontSave | HideFlags.NotEditable;
}
void CheckAndRebuildInstances()
{
bool rebuild = m_Instances.Length != m_InstanceCount;
#if UNITY_EDITOR
if (m_Dirty)
{
rebuild = true;
m_Dirty = false;
}
#endif
if (rebuild)
{
DisposeInstances();
if (m_PrefabToSpawn != null && m_InstanceCount != 0)
{
m_Instances = new GameObject[m_InstanceCount];
m_TimesToLive = new float[m_InstanceCount];
#if UNITY_EDITOR
var prefabAssetType = UnityEditor.PrefabUtility.GetPrefabAssetType(m_PrefabToSpawn);
#endif
for (int i = 0; i < m_Instances.Length; i++)
{
GameObject newInstance = null;
#if UNITY_EDITOR
if (prefabAssetType != UnityEditor.PrefabAssetType.NotAPrefab)
newInstance = UnityEditor.PrefabUtility.InstantiatePrefab(m_PrefabToSpawn) as GameObject;
if (newInstance == null)
newInstance = Instantiate(m_PrefabToSpawn);
#else
newInstance = Instantiate(m_PrefabToSpawn);
#endif
newInstance.name = $"{name} - #{i} - {m_PrefabToSpawn.name}";
newInstance.SetActive(false);
newInstance.transform.parent = m_ParentInstances ? transform : null;
UpdateHideFlag(newInstance);
m_Instances[i] = newInstance;
m_TimesToLive[i] = float.NegativeInfinity;
}
}
}
}
public override void OnVFXOutputEvent(VFXEventAttribute eventAttribute)
{
CheckAndRebuildInstances();
int freeIdx = -1;
for (int i = 0; i < m_Instances.Length; i++)
{
if (!m_Instances[i].activeSelf)
{
freeIdx = i;
break;
}
}
if (freeIdx != -1)
{
var availableInstance = m_Instances[freeIdx];
availableInstance.SetActive(true);
if (usePosition && eventAttribute.HasVector3(k_PositionID))
{
if (m_ParentInstances)
availableInstance.transform.localPosition = eventAttribute.GetVector3(k_PositionID);
else
availableInstance.transform.position = eventAttribute.GetVector3(k_PositionID);
}
if (useAngle && eventAttribute.HasVector3(k_AngleID))
{
if (parentInstances)
availableInstance.transform.localEulerAngles = eventAttribute.GetVector3(k_AngleID);
else
availableInstance.transform.eulerAngles = eventAttribute.GetVector3(k_AngleID);
}
if (useScale && eventAttribute.HasVector3(k_ScaleID))
availableInstance.transform.localScale = eventAttribute.GetVector3(k_ScaleID);
if (useLifetime && eventAttribute.HasFloat(k_LifetimeID))
m_TimesToLive[freeIdx] = eventAttribute.GetFloat(k_LifetimeID);
else
m_TimesToLive[freeIdx] = float.NegativeInfinity;
var handlers = availableInstance.GetComponentsInChildren<VFXOutputEventPrefabAttributeAbstractHandler>();
foreach (var handler in handlers)
handler.OnVFXEventAttribute(eventAttribute, m_VisualEffect);
} //Else, can't find an instance available, ignoring.
}
void Update()
{
if (Application.isPlaying || (executeInEditor && canExecuteInEditor))
{
CheckAndRebuildInstances();
var dt = Time.deltaTime;
for (int i = 0; i < m_Instances.Length; i++)
{
#if UNITY_EDITOR
//Reassign hide flag, "open prefab" could have resetted this hide flag.
UpdateHideFlag(m_Instances[i]);
#endif
// Negative infinity for non-time managed
if (m_TimesToLive[i] == float.NegativeInfinity)
continue;
// Else, manage time
if (m_TimesToLive[i] <= 0.0f && m_Instances[i].activeSelf)
m_Instances[i].SetActive(false);
else
m_TimesToLive[i] -= dt;
}
}
else
{
DisposeInstances();
}
}
}
}