217 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			217 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | /******************************************************************************/ | |||
|  | /* | |||
|  |   Project   - MudBun | |||
|  |   Publisher - Long Bunny Labs | |||
|  |               http://LongBunnyLabs.com | |||
|  |   Author    - Ming-Lun "Allen" Chou | |||
|  |               http://AllenChou.net | |||
|  | */ | |||
|  | /******************************************************************************/ | |||
|  | 
 | |||
|  | using System.Collections.Generic; | |||
|  | 
 | |||
|  | using Unity.Collections; | |||
|  | using UnityEngine; | |||
|  | 
 | |||
|  | #if MUDBUN_BURST | |||
|  | using Unity.Burst; | |||
|  | using Unity.Mathematics; | |||
|  | using AOT; | |||
|  | #endif | |||
|  | 
 | |||
|  | namespace MudBun | |||
|  | { | |||
|  | #if MUDBUN_BURST | |||
|  |   [BurstCompile] | |||
|  | #endif | |||
|  |   [ExecuteInEditMode] | |||
|  |   public class MudParticleSystem : MudSolid | |||
|  |   { | |||
|  |     public ParticleSystem Particles; | |||
|  | 
 | |||
|  |     [SerializeField] private float m_radiusMultiplier = 1.0f; | |||
|  |     public float RadiusMultiplier { get => m_radiusMultiplier; set { m_radiusMultiplier = value; MarkDirty(); } } | |||
|  | 
 | |||
|  |     [SerializeField] private float m_selfBlend = 0.5f; | |||
|  |     public float SelfBlend { get => m_selfBlend; set { m_selfBlend = value; MarkDirty(); } } | |||
|  | 
 | |||
|  |     private static readonly int InitNumParticles = 16; | |||
|  |     private int m_lastReadFrame = -1; | |||
|  | 
 | |||
|  |     // read particle data | |||
|  |     private ParticleSystem.Particle [] m_aParticle = new ParticleSystem.Particle[InitNumParticles]; | |||
|  |     private int m_numParticles = 0; | |||
|  |     private Vector3 [] m_aPosWs = new Vector3[InitNumParticles]; | |||
|  |     private Vector3 [] m_aPosRs = new Vector3[InitNumParticles]; | |||
|  |     private float [] m_aRadius = new float[InitNumParticles]; | |||
|  |     private float [] m_aSelfBlendMult = new float[InitNumParticles]; | |||
|  | 
 | |||
|  |     public override Aabb RawBoundsRs | |||
|  |     { | |||
|  |       get | |||
|  |       { | |||
|  |         if (!ReadParticles()) | |||
|  |           return Aabb.Empty; | |||
|  | 
 | |||
|  |         Aabb bounds = Aabb.Empty; | |||
|  |         for (int i = 0, n = Particles.particleCount; i < n; ++i) | |||
|  |         { | |||
|  |           Vector3 posCs = m_aPosRs[i]; | |||
|  |           Vector3 r = (m_aRadius[i] + m_selfBlend) * Vector3.one; | |||
|  |           bounds.Include(new Aabb(posCs - r, posCs + r)); | |||
|  |         } | |||
|  | 
 | |||
|  |         return bounds; | |||
|  |       } | |||
|  |     } | |||
|  | 
 | |||
|  |     private bool ReadParticles() | |||
|  |     { | |||
|  |       if (Particles == null) | |||
|  |         return false; | |||
|  | 
 | |||
|  |       if (!Particles.isPlaying) | |||
|  |         return m_numParticles > 0; | |||
|  | 
 | |||
|  |       if (m_lastReadFrame >= Time.renderedFrameCount) | |||
|  |         return m_aParticle.Length >= Particles.particleCount; | |||
|  | 
 | |||
|  |       if (m_aParticle.Length < Particles.particleCount) | |||
|  |       { | |||
|  |         int newLen = m_aParticle.Length; | |||
|  |         while (newLen < Particles.particleCount) | |||
|  |           newLen *= 2; | |||
|  | 
 | |||
|  |         m_aParticle = new ParticleSystem.Particle[newLen]; | |||
|  |         m_aPosWs = new Vector3[newLen]; | |||
|  |         m_aPosRs = new Vector3[newLen]; | |||
|  |         m_aRadius = new float[newLen]; | |||
|  |         m_aSelfBlendMult = new float[newLen]; | |||
|  |       } | |||
|  | 
 | |||
|  |       float selfBlendSizeFactor = Mathf.Clamp(5.0f / Mathf.Max(MathUtil.Epsilon, Particles.main.startSizeMultiplier), 0.1f, 100.0f); | |||
|  |       Particles.GetParticles(m_aParticle, Particles.particleCount); | |||
|  |       for (int i = 0, n = Particles.particleCount; i < n; ++i) | |||
|  |       { | |||
|  |         m_aPosWs[i] = Particles.gameObject.transform.TransformPoint(m_aParticle[i].position); | |||
|  |         m_aRadius[i] = m_aParticle[i].GetCurrentSize(Particles) * m_radiusMultiplier; | |||
|  |         m_aSelfBlendMult[i] = Mathf.Clamp01(m_aParticle[i].GetCurrentSize(Particles) * selfBlendSizeFactor); | |||
|  |       } | |||
|  | 
 | |||
|  |       switch (Particles.main.simulationSpace) | |||
|  |       { | |||
|  |         case ParticleSystemSimulationSpace.Local: | |||
|  |           break; | |||
|  |         case ParticleSystemSimulationSpace.World: | |||
|  |           for (int i = 0, n = Particles.particleCount; i < n; ++i) | |||
|  |             m_aPosWs[i] = transform.InverseTransformPoint(m_aPosWs[i]); | |||
|  |           break; | |||
|  |         case ParticleSystemSimulationSpace.Custom: | |||
|  |           if (Particles.main.customSimulationSpace != null) | |||
|  |             for (int i = 0, n = Particles.particleCount; i < n; ++i) | |||
|  |               m_aPosWs[i] = Particles.main.customSimulationSpace.InverseTransformPoint(m_aPosWs[i]); | |||
|  |           break; | |||
|  |       } | |||
|  | 
 | |||
|  |       for (int i = 0, n = Particles.particleCount; i < n; ++i) | |||
|  |         m_aPosRs[i] = PointRs(m_aPosWs[i]); | |||
|  | 
 | |||
|  |       m_lastReadFrame = Time.renderedFrameCount; | |||
|  |       m_numParticles = Particles.particleCount; | |||
|  |       return true; | |||
|  |     } | |||
|  | 
 | |||
|  |     private void LateUpdate() | |||
|  |     { | |||
|  |       if (!ReadParticles()) | |||
|  |         return; | |||
|  | 
 | |||
|  |       MarkDirty(); | |||
|  |     } | |||
|  | 
 | |||
|  |     public override void SanitizeParameters() | |||
|  |     { | |||
|  |       base.SanitizeParameters(); | |||
|  | 
 | |||
|  |       Validate.NonNegative(ref m_radiusMultiplier); | |||
|  |       Validate.NonNegative(ref m_selfBlend); | |||
|  |     } | |||
|  | 
 | |||
|  |     private int m_iStart = -1; | |||
|  | 
 | |||
|  |     public override int FillComputeData(NativeArray<SdfBrush> aBrush, int iStart, List<Transform> aBone) | |||
|  |     { | |||
|  |       m_iStart = -1; | |||
|  | 
 | |||
|  |       if (!ReadParticles()) | |||
|  |         return 0; | |||
|  | 
 | |||
|  |       m_iStart = iStart; | |||
|  | 
 | |||
|  |       SdfBrush brush = SdfBrush.New(); | |||
|  | 
 | |||
|  |       if (aBone != null) | |||
|  |       { | |||
|  |         brush.BoneIndex = aBone.Count; | |||
|  |         aBone.Add(gameObject.transform); | |||
|  |       } | |||
|  | 
 | |||
|  |       for (int i = 0, n = m_numParticles; i < n; ++i) | |||
|  |       { | |||
|  |         if (i == 0) | |||
|  |         { | |||
|  |           brush.Type = (int) SdfBrush.TypeEnum.ParticleSystem; | |||
|  |           brush.Data2.x = n; | |||
|  |         } | |||
|  |         else | |||
|  |         { | |||
|  |           brush.Type = (int) SdfBrush.TypeEnum.Nop; | |||
|  |         } | |||
|  | 
 | |||
|  |         Vector3 posCs = m_aPosRs[i]; | |||
|  |         float r = m_aRadius[i]; | |||
|  |         brush.Data0 = new Vector4(posCs.x, posCs.y, posCs.z, r); | |||
|  | 
 | |||
|  |         // fade out self blend as particles die off to avoid pops | |||
|  |         brush.Data1.x = m_selfBlend * Mathf.Clamp01(10.0f * (m_aSelfBlendMult[i] - 0.1f)); | |||
|  | 
 | |||
|  |         aBrush[iStart++] = brush; | |||
|  |       } | |||
|  | 
 | |||
|  |       return iStart - m_iStart; | |||
|  |     } | |||
|  | 
 | |||
|  |     public override void FillBrushData(ref SdfBrush brush, int iBrush) | |||
|  |     { | |||
|  |       if (m_iStart < 0) | |||
|  |         return; | |||
|  | 
 | |||
|  |       // only need to fill in the first one | |||
|  |       if (iBrush == m_iStart) | |||
|  |         base.FillBrushData(ref brush, iBrush); | |||
|  | 
 | |||
|  |       brush.Position = m_aPosWs[brush.Index - m_iStart]; | |||
|  |     } | |||
|  | 
 | |||
|  | #if MUDBUN_BURST | |||
|  |     [BurstCompile] | |||
|  |     [MonoPInvokeCallback(typeof(Sdf.SdfBrushEvalFunc))] | |||
|  |     [RegisterSdfBrushEvalFunc(SdfBrush.TypeEnum.ParticleSystem)] | |||
|  |     public static unsafe float EvaluateSdf(float resDummy, ref float3 p, in float3 pRel, SdfBrush* aBrush, int iBrush) | |||
|  |     { | |||
|  |       float res = float.MaxValue; | |||
|  |       int numParticles = (int) aBrush[iBrush].Data2.x; | |||
|  |       for (int i = 0; i < numParticles; ++i) | |||
|  |       { | |||
|  |         float3 pos = new float4(aBrush[iBrush + i].Data0).xyz; | |||
|  |         float r = aBrush[iBrush + i].Data0.w; | |||
|  |         float selfBlend = aBrush[iBrush + i].Data1.x; | |||
|  |         res = Sdf.UniCubic(res, Sdf.Sphere(p - pos, r), selfBlend); | |||
|  |       } | |||
|  |       return res; | |||
|  |     } | |||
|  | #endif | |||
|  |   } | |||
|  | } | |||
|  | 
 |