From 78fbff1ebe4b14db4488d53e1910b577137daa08 Mon Sep 17 00:00:00 2001 From: dogramacigokhan Date: Sun, 7 Jul 2024 11:05:42 +0300 Subject: [PATCH] feat: Support list of particles in ParticleAttractor instead of a single one. --- Packages/src/Runtime/UIParticleAttractor.cs | 169 +++++++++++--------- 1 file changed, 97 insertions(+), 72 deletions(-) diff --git a/Packages/src/Runtime/UIParticleAttractor.cs b/Packages/src/Runtime/UIParticleAttractor.cs index 56c9e7e1..b165cf71 100644 --- a/Packages/src/Runtime/UIParticleAttractor.cs +++ b/Packages/src/Runtime/UIParticleAttractor.cs @@ -1,4 +1,5 @@ using System; +using System.Collections.Generic; using Coffee.UIParticleExtensions; using UnityEngine; using UnityEngine.Events; @@ -22,7 +23,7 @@ public enum UpdateMode } [SerializeField] - private ParticleSystem m_ParticleSystem; + private List m_ParticleSystems; [Range(0.1f, 10f)] [SerializeField] @@ -45,7 +46,7 @@ public enum UpdateMode [SerializeField] private UnityEvent m_OnAttracted; - private UIParticle _uiParticle; + private UIParticle[] _uiParticles; public float destinationRadius { @@ -84,25 +85,42 @@ public UnityEvent onAttracted } /// - /// The target ParticleSystem to attract. + /// The target ParticleSystems to attract. Use and + /// to modify the list. /// -#if UNITY_EDITOR - public new ParticleSystem particleSystem -#else - public ParticleSystem particleSystem -#endif + public IReadOnlyList particleSystems => m_ParticleSystems; + + public void AddParticleSystem(ParticleSystem ps) + { + if (m_ParticleSystems == null) + { + m_ParticleSystems = new List(); + } + + if (!m_ParticleSystems.Contains(ps)) + { + m_ParticleSystems.Add(ps); + ApplyParticleSystems(); + } + } + + public void RemoveParticleSystem(ParticleSystem ps) { - get => m_ParticleSystem; - set + if (m_ParticleSystems == null) { - m_ParticleSystem = value; - ApplyParticleSystem(); + return; + } + + if (m_ParticleSystems.Contains(ps)) + { + m_ParticleSystems.Remove(ps); + ApplyParticleSystems(); } } private void OnEnable() { - ApplyParticleSystem(); + ApplyParticleSystems(); UIParticleUpdater.Register(this); } @@ -113,85 +131,95 @@ private void OnDisable() private void OnDestroy() { - _uiParticle = null; - m_ParticleSystem = null; + _uiParticles = null; + m_ParticleSystems = null; } internal void Attract() { - if (m_ParticleSystem == null) return; + if (m_ParticleSystems == null) return; - var count = m_ParticleSystem.particleCount; - if (count == 0) return; + for (var particleIndex = 0; particleIndex < this.m_ParticleSystems.Count; particleIndex++) + { + var particleSystem = m_ParticleSystems[particleIndex]; + if (particleSystem == null || !particleSystem.gameObject.activeInHierarchy) continue; - var particles = ParticleSystemExtensions.GetParticleArray(count); - m_ParticleSystem.GetParticles(particles, count); + var count = particleSystem.particleCount; + if (count == 0) continue; - var dstPos = GetDestinationPosition(); - for (var i = 0; i < count; i++) - { - // Attracted - var p = particles[i]; - if (0f < p.remainingLifetime && Vector3.Distance(p.position, dstPos) < m_DestinationRadius) - { - p.remainingLifetime = 0f; - particles[i] = p; + var particles = ParticleSystemExtensions.GetParticleArray(count); + particleSystem.GetParticles(particles, count); + + var uiParticle = this._uiParticles != null && particleIndex < _uiParticles.Length + ? _uiParticles[particleIndex] + : null; - if (m_OnAttracted != null) + var dstPos = this.GetDestinationPosition(uiParticle, particleSystem); + for (var i = 0; i < count; i++) + { + // Attracted + var p = particles[i]; + if (0f < p.remainingLifetime && Vector3.Distance(p.position, dstPos) < this.m_DestinationRadius) { - try - { - m_OnAttracted.Invoke(); - } - catch (Exception e) + p.remainingLifetime = 0f; + particles[i] = p; + + if (this.m_OnAttracted != null) { - Debug.LogException(e); + try + { + this.m_OnAttracted.Invoke(); + } + catch (Exception e) + { + Debug.LogException(e); + } } + + continue; } - continue; - } + // Calc attracting time + var delayTime = p.startLifetime * this.m_DelayRate; + var duration = p.startLifetime - delayTime; + var time = Mathf.Max(0, p.startLifetime - p.remainingLifetime - delayTime); - // Calc attracting time - var delayTime = p.startLifetime * m_DelayRate; - var duration = p.startLifetime - delayTime; - var time = Mathf.Max(0, p.startLifetime - p.remainingLifetime - delayTime); + // Delay + if (time <= 0) continue; - // Delay - if (time <= 0) continue; + // Attract + p.position = this.GetAttractedPosition(p.position, dstPos, duration, time); + p.velocity *= 0.5f; + particles[i] = p; + } - // Attract - p.position = GetAttractedPosition(p.position, dstPos, duration, time); - p.velocity *= 0.5f; - particles[i] = p; + particleSystem.SetParticles(particles, count); } - - m_ParticleSystem.SetParticles(particles, count); } - private Vector3 GetDestinationPosition() + private Vector3 GetDestinationPosition(UIParticle uiParticle, ParticleSystem particleSystem) { - var isUI = _uiParticle && _uiParticle.enabled; - var psPos = m_ParticleSystem.transform.position; + var isUI = uiParticle && uiParticle.enabled; + var psPos = particleSystem.transform.position; var attractorPos = transform.position; var dstPos = attractorPos; - var isLocalSpace = m_ParticleSystem.IsLocalSpace(); + var isLocalSpace = particleSystem.IsLocalSpace(); if (isLocalSpace) { - dstPos = m_ParticleSystem.transform.InverseTransformPoint(dstPos); + dstPos = particleSystem.transform.InverseTransformPoint(dstPos); } if (isUI) { - var inverseScale = _uiParticle.parentScale.Inverse(); - var scale3d = _uiParticle.scale3DForCalc; + var inverseScale = uiParticle.parentScale.Inverse(); + var scale3d = uiParticle.scale3DForCalc; dstPos = dstPos.GetScaled(inverseScale, scale3d.Inverse()); // Relative mode - if (_uiParticle.positionMode == UIParticle.PositionMode.Relative) + if (uiParticle.positionMode == UIParticle.PositionMode.Relative) { - var diff = _uiParticle.transform.position - psPos; + var diff = uiParticle.transform.position - psPos; diff.Scale(scale3d - inverseScale); diff.Scale(scale3d.Inverse()); dstPos += diff; @@ -237,25 +265,22 @@ private Vector3 GetAttractedPosition(Vector3 current, Vector3 target, float dura return Vector3.MoveTowards(current, target, speed); } - private void ApplyParticleSystem() + private void ApplyParticleSystems() { - _uiParticle = null; - if (m_ParticleSystem == null) + _uiParticles = null; + if (m_ParticleSystems == null || m_ParticleSystems.Count == 0) { -#if UNITY_EDITOR - if (Application.isPlaying) -#endif - { - Debug.LogError("No particle system attached to particle attractor script", this); - } - return; } - _uiParticle = m_ParticleSystem.GetComponentInParent(true); - if (_uiParticle && !_uiParticle.particles.Contains(m_ParticleSystem)) + _uiParticles = new UIParticle[m_ParticleSystems.Count]; + for (var i = 0; i < this.m_ParticleSystems.Count; i++) { - _uiParticle = null; + var particleSystem = m_ParticleSystems[i]; + if (particleSystem == null) continue; + + var uiParticle = particleSystem.GetComponentInParent(true); + _uiParticles[i] = uiParticle.particles.Contains(particleSystem) ? uiParticle : null; } } }