using System; using System.Collections.Generic; using Coffee.UIParticleExtensions; using UnityEditor; using UnityEngine; using UnityEngine.Profiling; using UnityEngine.Rendering; using UnityEngine.UI; namespace Coffee.UIExtensions { [ExecuteAlways] [RequireComponent(typeof(RectTransform))] [RequireComponent(typeof(CanvasRenderer))] [AddComponentMenu("")] internal class UIParticleRenderer : MaskableGraphic { private static readonly List<Component> s_Components = new List<Component>(); private static readonly CombineInstance[] s_CombineInstances = { new CombineInstance() }; private static readonly List<Material> s_Materials = new List<Material>(2); private static MaterialPropertyBlock s_Mpb; private static readonly List<UIParticleRenderer> s_Renderers = new List<UIParticleRenderer>(); private static readonly List<Color32> s_Colors = new List<Color32>(); private static readonly Vector3[] s_Corners = new Vector3[4]; private Material _currentMaterialForRendering; private bool _delay; private int _index; private bool _isTrail; private Bounds _lastBounds; private Material _modifiedMaterial; private UIParticle _parent; private ParticleSystem _particleSystem; private int _prevParticleCount; private Vector3 _prevPsPos; private Vector3 _prevScale; private Vector2Int _prevScreenSize; private float _prevCanvasScale; private bool _prewarm; private ParticleSystemRenderer _renderer; public override Texture mainTexture { get { return _isTrail ? null : _particleSystem.GetTextureForSprite(); } } public override bool raycastTarget { get { return false; } } private Rect rootCanvasRect { get { s_Corners[0] = transform.TransformPoint(_lastBounds.min.x, _lastBounds.min.y, 0); s_Corners[1] = transform.TransformPoint(_lastBounds.min.x, _lastBounds.max.y, 0); s_Corners[2] = transform.TransformPoint(_lastBounds.max.x, _lastBounds.max.y, 0); s_Corners[3] = transform.TransformPoint(_lastBounds.max.x, _lastBounds.min.y, 0); if (canvas) { var worldToLocalMatrix = canvas.rootCanvas.transform.worldToLocalMatrix; for (var i = 0; i < 4; ++i) { s_Corners[i] = worldToLocalMatrix.MultiplyPoint(s_Corners[i]); } } var corner1 = (Vector2)s_Corners[0]; var corner2 = (Vector2)s_Corners[0]; for (var i = 1; i < 4; ++i) { if (s_Corners[i].x < corner1.x) { corner1.x = s_Corners[i].x; } else if (s_Corners[i].x > corner2.x) { corner2.x = s_Corners[i].x; } if (s_Corners[i].y < corner1.y) { corner1.y = s_Corners[i].y; } else if (s_Corners[i].y > corner2.y) { corner2.y = s_Corners[i].y; } } return new Rect(corner1, corner2 - corner1); } } public void Reset(int index = -1) { if (_renderer) { _renderer.enabled = true; } _parent = null; _particleSystem = null; _renderer = null; _prevParticleCount = 0; if (0 <= index) { _index = index; } //_emitter = null; if (this && isActiveAndEnabled) { material = null; workerMesh.Clear(); canvasRenderer.SetMesh(workerMesh); _lastBounds = new Bounds(); enabled = false; } else { ModifiedMaterial.Remove(_modifiedMaterial); _modifiedMaterial = null; _currentMaterialForRendering = null; } } protected override void OnEnable() { base.OnEnable(); if (!s_CombineInstances[0].mesh) { s_CombineInstances[0].mesh = new Mesh { name = "[UIParticleRenderer] Combine Instance Mesh", hideFlags = HideFlags.HideAndDontSave }; } _currentMaterialForRendering = null; } protected override void OnDisable() { base.OnDisable(); ModifiedMaterial.Remove(_modifiedMaterial); _modifiedMaterial = null; _currentMaterialForRendering = null; } public static UIParticleRenderer AddRenderer(UIParticle parent, int index) { // Create renderer object. var go = new GameObject("[generated] UIParticleRenderer", typeof(UIParticleRenderer)) { hideFlags = HideFlags.DontSave | HideFlags.NotEditable, layer = parent.gameObject.layer }; // Set parent. var transform = go.transform; transform.SetParent(parent.transform, false); transform.localPosition = Vector3.zero; transform.localRotation = Quaternion.identity; transform.localScale = Vector3.one; // Add renderer component. var renderer = go.GetComponent<UIParticleRenderer>(); renderer._parent = parent; renderer._index = index; return renderer; } /// <summary> /// Perform material modification in this function. /// </summary> public override Material GetModifiedMaterial(Material baseMaterial) { _currentMaterialForRendering = null; if (!IsActive() || !_parent) { ModifiedMaterial.Remove(_modifiedMaterial); _modifiedMaterial = null; return baseMaterial; } var modifiedMaterial = base.GetModifiedMaterial(baseMaterial); // var texture = mainTexture; if (texture == null && _parent.m_AnimatableProperties.Length == 0) { ModifiedMaterial.Remove(_modifiedMaterial); _modifiedMaterial = null; return modifiedMaterial; } // var id = _parent.m_AnimatableProperties.Length == 0 ? 0 : GetInstanceID(); #if UNITY_EDITOR var props = EditorJsonUtility.ToJson(modifiedMaterial).GetHashCode(); #else var props = 0; #endif modifiedMaterial = ModifiedMaterial.Add(modifiedMaterial, texture, id, props); ModifiedMaterial.Remove(_modifiedMaterial); _modifiedMaterial = modifiedMaterial; return modifiedMaterial; } public void Set(UIParticle parent, ParticleSystem ps, bool isTrail) { _parent = parent; maskable = parent.maskable; gameObject.layer = parent.gameObject.layer; _particleSystem = ps; _prewarm = _particleSystem.main.prewarm; #if UNITY_EDITOR if (Application.isPlaying) #endif { if (_particleSystem.isPlaying || _prewarm) { _particleSystem.Clear(); _particleSystem.Pause(); } } _renderer = ps.GetComponent<ParticleSystemRenderer>(); _renderer.enabled = false; //_emitter = emitter; _isTrail = isTrail; _renderer.GetSharedMaterials(s_Materials); material = s_Materials[isTrail ? 1 : 0]; s_Materials.Clear(); // Support sprite. var tsa = ps.textureSheetAnimation; if (tsa.mode == ParticleSystemAnimationMode.Sprites && tsa.uvChannelMask == 0) { tsa.uvChannelMask = UVChannelFlags.UV0; } _prevScale = GetWorldScale(); _prevPsPos = _particleSystem.transform.position; _prevScreenSize = new Vector2Int(Screen.width, Screen.height); _prevCanvasScale = canvas ? canvas.scaleFactor : 1f; _delay = true; _prevParticleCount = 0; canvasRenderer.SetTexture(null); enabled = true; } public void UpdateMesh(Camera bakeCamera) { // No particle to render: Clear mesh. if ( !isActiveAndEnabled || !_particleSystem || !_parent || !canvasRenderer || !canvas || !bakeCamera || _parent.meshSharing == UIParticle.MeshSharing.Replica || !transform.lossyScale.GetScaled(_parent.scale3DForCalc).IsVisible() // Scale is not visible. || (!_particleSystem.IsAlive() && !_particleSystem.isPlaying) // No particle. || (_isTrail && !_particleSystem.trails.enabled) // Trail, but it is not enabled. #if UNITY_2018_3_OR_NEWER || canvasRenderer.GetInheritedAlpha() < 0.01f // #102: Do not bake particle system to mesh when the alpha is zero. #endif ) { Profiler.BeginSample("[UIParticleRenderer] Clear Mesh"); workerMesh.Clear(); canvasRenderer.SetMesh(workerMesh); _lastBounds = new Bounds(); Profiler.EndSample(); return; } var main = _particleSystem.main; var scale = GetWorldScale(); var psPos = _particleSystem.transform.position; // Simulate particles. Profiler.BeginSample("[UIParticle] Bake Mesh > Simulate Particles"); if (!_isTrail && _parent.canSimulate) { #if UNITY_EDITOR if (!Application.isPlaying) { SimulateForEditor(psPos - _prevPsPos, scale); } else #endif { ResolveResolutionChange(psPos, scale); Simulate(scale, _parent.isPaused || _delay); if (_delay && !_parent.isPaused) { Simulate(scale, _parent.isPaused); } // When the ParticleSystem simulation is complete, stop it. if (!main.loop && main.duration <= _particleSystem.time && (_particleSystem.IsAlive() || _particleSystem.particleCount == 0) ) { _particleSystem.Stop(false); } } _prevScale = scale; _prevPsPos = psPos; _delay = false; } Profiler.EndSample(); // Bake mesh. Profiler.BeginSample("[UIParticleRenderer] Bake Mesh"); if (_isTrail && _parent.canSimulate && 0 < s_CombineInstances[0].mesh.vertexCount) { _renderer.BakeTrailsMesh(s_CombineInstances[0].mesh, bakeCamera, true); } else if (_renderer.CanBakeMesh()) { _renderer.BakeMesh(s_CombineInstances[0].mesh, bakeCamera, true); } else { s_CombineInstances[0].mesh.Clear(false); } // Too many vertices to render. if (65535 <= s_CombineInstances[0].mesh.vertexCount) { s_CombineInstances[0].mesh.Clear(false); Debug.LogErrorFormat(this, "Too many vertices to render. index={0}, isTrail={1}, vertexCount={2}(>=65535)", _index, _isTrail, s_CombineInstances[0].mesh.vertexCount ); s_CombineInstances[0].mesh.Clear(false); } Profiler.EndSample(); // Combine mesh to transform. ([ParticleSystem local ->] world -> renderer local) Profiler.BeginSample("[UIParticleRenderer] Combine Mesh"); if (_parent.canSimulate) { if (_parent.positionMode == UIParticle.PositionMode.Absolute) { s_CombineInstances[0].transform = canvasRenderer.transform.worldToLocalMatrix * GetWorldMatrix(psPos, scale); } else { var diff = _particleSystem.transform.position - _parent.transform.position; s_CombineInstances[0].transform = canvasRenderer.transform.worldToLocalMatrix * Matrix4x4.Translate(diff.GetScaled(scale - Vector3.one)) * GetWorldMatrix(psPos, scale); } workerMesh.CombineMeshes(s_CombineInstances, true, true); workerMesh.RecalculateBounds(); var bounds = workerMesh.bounds; var center = bounds.center; center.z = 0; bounds.center = center; var extents = bounds.extents; extents.z = 0; bounds.extents = extents; workerMesh.bounds = bounds; _lastBounds = bounds; // Convert linear color to gamma color. if (QualitySettings.activeColorSpace == ColorSpace.Linear) { Profiler.BeginSample("[UIParticleRenderer] Convert Linear to Gamma"); workerMesh.GetColors(s_Colors); var count_c = s_Colors.Count; for (var i = 0; i < count_c; i++) { var c = s_Colors[i]; c.r = c.r.LinearToGamma(); c.g = c.g.LinearToGamma(); c.b = c.b.LinearToGamma(); s_Colors[i] = c; } workerMesh.SetColors(s_Colors); Profiler.EndSample(); } GetComponents(typeof(IMeshModifier), s_Components); for (var i = 0; i < s_Components.Count; i++) { #pragma warning disable CS0618 // Type or member is obsolete ((IMeshModifier)s_Components[i]).ModifyMesh(workerMesh); #pragma warning restore CS0618 // Type or member is obsolete } s_Components.Clear(); } Profiler.EndSample(); // Get grouped renderers. s_Renderers.Clear(); if (_parent.useMeshSharing) { UIParticleUpdater.GetGroupedRenderers(_parent.groupId, _index, s_Renderers); } // Set mesh to the CanvasRenderer. Profiler.BeginSample("[UIParticleRenderer] Set Mesh"); for (var i = 0; i < s_Renderers.Count; i++) { if (s_Renderers[i] == this) continue; s_Renderers[i].canvasRenderer.SetMesh(workerMesh); s_Renderers[i]._lastBounds = _lastBounds; } if (!_parent.canRender) { workerMesh.Clear(); } canvasRenderer.SetMesh(workerMesh); Profiler.EndSample(); // Update animatable material properties. Profiler.BeginSample("[UIParticleRenderer] Update Animatable Material Properties"); #if UNITY_EDITOR if (_modifiedMaterial != material) { _renderer.GetSharedMaterials(s_Materials); material = s_Materials[_isTrail ? 1 : 0]; s_Materials.Clear(); SetMaterialDirty(); } #endif UpdateMaterialProperties(); if (_parent.useMeshSharing) { if (!_currentMaterialForRendering) { _currentMaterialForRendering = materialForRendering; } for (var i = 0; i < s_Renderers.Count; i++) { if (s_Renderers[i] == this) continue; s_Renderers[i].canvasRenderer.materialCount = 1; s_Renderers[i].canvasRenderer.SetMaterial(_currentMaterialForRendering, 0); } } Profiler.EndSample(); s_Renderers.Clear(); } internal void UpdateParticleCount() { if (!_particleSystem) return; _prevParticleCount = _particleSystem.particleCount; } /// <summary> /// Call to update the geometry of the Graphic onto the CanvasRenderer. /// </summary> protected override void UpdateGeometry() { } public override void Cull(Rect clipRect, bool validRect) { var cull = _lastBounds.extents == Vector3.zero || !validRect || !clipRect.Overlaps(rootCanvasRect, true); if (canvasRenderer.cull == cull) return; canvasRenderer.cull = cull; UISystemProfilerApi.AddMarker("MaskableGraphic.cullingChanged", this); onCullStateChanged.Invoke(cull); OnCullingChanged(); } private Vector3 GetWorldScale() { Profiler.BeginSample("[UIParticleRenderer] GetWorldScale"); var scale = _parent.scale3DForCalc.GetScaled(_parent.parentScale); Profiler.EndSample(); return scale; } private Matrix4x4 GetWorldMatrix(Vector3 psPos, Vector3 scale) { var space = _particleSystem.GetActualSimulationSpace(); if (_isTrail && _particleSystem.trails.worldSpace) { space = ParticleSystemSimulationSpace.World; } #if UNITY_EDITOR if (!Application.isPlaying) { switch (space) { case ParticleSystemSimulationSpace.World: return Matrix4x4.Translate(psPos) * Matrix4x4.Scale(scale) * Matrix4x4.Translate(-psPos); } } #endif switch (space) { case ParticleSystemSimulationSpace.Local: return Matrix4x4.Translate(psPos) * Matrix4x4.Scale(scale); case ParticleSystemSimulationSpace.World: return Matrix4x4.Scale(scale); case ParticleSystemSimulationSpace.Custom: return Matrix4x4.Translate(_particleSystem.main.customSimulationSpace.position.GetScaled(scale)) //* Matrix4x4.Translate(wpos) * Matrix4x4.Scale(scale) //* Matrix4x4.Translate(-wpos) ; default: throw new NotSupportedException(); } } /// <summary> /// For world simulation, interpolate particle positions when the screen size is changed. /// </summary> /// <param name="psPos"></param> /// <param name="scale"></param> private void ResolveResolutionChange(Vector3 psPos, Vector3 scale) { var screenSize = new Vector2Int(Screen.width, Screen.height); var isWorldSpace = _particleSystem.IsWorldSpace(); var canvasScale = _parent.canvas ? _parent.canvas.scaleFactor : 1f; var resolutionChanged = _prevScreenSize != screenSize || _prevCanvasScale != canvasScale; if (resolutionChanged && isWorldSpace) { // Update particle array size and get particles. var size = _particleSystem.particleCount; var particles = ParticleSystemExtensions.GetParticleArray(size); _particleSystem.GetParticles(particles, size); // Resolusion resolver: // (psPos / scale) / (prevPsPos / prevScale) -> psPos * scale.inv * prevPsPos.inv * prevScale var modifier = psPos.GetScaled( scale.Inverse(), _prevPsPos.Inverse(), _prevScale); for (var i = 0; i < size; i++) { var particle = particles[i]; particle.position = particle.position.GetScaled(modifier); particles[i] = particle; } _particleSystem.SetParticles(particles, size); // Delay: Do not progress in the frame where the resolution has been changed. _delay = true; _prevScale = scale; _prevPsPos = psPos; } _prevCanvasScale = canvas ? canvas.scaleFactor : 1f; _prevScreenSize = screenSize; } private void Simulate(Vector3 scale, bool paused) { var main = _particleSystem.main; var deltaTime = paused ? 0 : main.useUnscaledTime ? Time.unscaledDeltaTime : Time.deltaTime; // Prewarm: if (0 < deltaTime && _prewarm) { deltaTime += main.duration; _prewarm = false; } // (COMMENT OUT) #231: Sub Emitters option is not work in editor playing /* // Emitted particles found. if (_prevParticleCount != _particleSystem.particleCount) { var size = _particleSystem.particleCount; var particles = ParticleSystemExtensions.GetParticleArray(size); _particleSystem.GetParticles(particles, size); for (var i = _prevParticleCount; i < size; i++) { var p = particles[i]; p.position = p.position.GetScaled(scale.Inverse()); particles[i] = p; } _particleSystem.SetParticles(particles, size); } */ // get world position. var isLocalSpace = _particleSystem.IsLocalSpace(); var psTransform = _particleSystem.transform; var originWorldPosition = psTransform.position; var originWorldRotation = psTransform.rotation; var emission = _particleSystem.emission; var rateOverDistance = emission.enabled && 0 < emission.rateOverDistance.constant && 0 < emission.rateOverDistanceMultiplier; if (rateOverDistance && !paused) { // (For rate-over-distance emission,) Move to previous scaled position, simulate (delta = 0). var prevScaledPos = isLocalSpace ? _prevPsPos : _prevPsPos.GetScaled(_prevScale.Inverse()); psTransform.SetPositionAndRotation(prevScaledPos, originWorldRotation); _particleSystem.Simulate(0, false, false, false); } // Move to scaled position, simulate, revert to origin position. var scaledPos = isLocalSpace ? originWorldPosition : originWorldPosition.GetScaled(scale.Inverse()); psTransform.SetPositionAndRotation(scaledPos, originWorldRotation); _particleSystem.Simulate(deltaTime, false, false, false); psTransform.SetPositionAndRotation(originWorldPosition, originWorldRotation); } #if UNITY_EDITOR private void SimulateForEditor(Vector3 diffPos, Vector3 scale) { // Extra world simulation. var isWorldSpace = _particleSystem.IsWorldSpace(); if (isWorldSpace && 0 < Vector3.SqrMagnitude(diffPos)) { Profiler.BeginSample("[UIParticle] Bake Mesh > Extra world simulation"); diffPos.x *= 1f - 1f / Mathf.Max(0.001f, scale.x); diffPos.y *= 1f - 1f / Mathf.Max(0.001f, scale.y); diffPos.z *= 1f - 1f / Mathf.Max(0.001f, scale.z); var size = _particleSystem.particleCount; var particles = ParticleSystemExtensions.GetParticleArray(size); _particleSystem.GetParticles(particles, size); for (var i = 0; i < size; i++) { var p = particles[i]; p.position += diffPos; particles[i] = p; } _particleSystem.SetParticles(particles, size); Profiler.EndSample(); } } #endif private void UpdateMaterialProperties() { if (_parent.m_AnimatableProperties.Length == 0) return; if (s_Mpb == null) { s_Mpb = new MaterialPropertyBlock(); } _renderer.GetPropertyBlock(s_Mpb); if (s_Mpb.isEmpty) return; // #41: Copy the value from MaterialPropertyBlock to CanvasRenderer if (!_modifiedMaterial) return; for (var i = 0; i < _parent.m_AnimatableProperties.Length; i++) { var ap = _parent.m_AnimatableProperties[i]; ap.UpdateMaterialProperties(_modifiedMaterial, s_Mpb); } s_Mpb.Clear(); } } }