diff --git a/Scripts/Editor/UIParticleEditor.cs b/Scripts/Editor/UIParticleEditor.cs index a37656d..9037d05 100644 --- a/Scripts/Editor/UIParticleEditor.cs +++ b/Scripts/Editor/UIParticleEditor.cs @@ -70,7 +70,7 @@ public override void OnGUI() private SerializedProperty _groupId; private SerializedProperty _groupMaxId; private SerializedProperty _positionMode; - private SerializedProperty _autoScaling; + private SerializedProperty _autoScalingMode; private ReorderableList _ro; private bool _showMax; @@ -153,7 +153,7 @@ protected override void OnEnable() _groupId = serializedObject.FindProperty("m_GroupId"); _groupMaxId = serializedObject.FindProperty("m_GroupMaxId"); _positionMode = serializedObject.FindProperty("m_PositionMode"); - _autoScaling = serializedObject.FindProperty("m_AutoScaling"); + _autoScalingMode = serializedObject.FindProperty("m_AutoScalingMode"); var sp = serializedObject.FindProperty("m_Particles"); _ro = new ReorderableList(sp.serializedObject, sp, true, true, true, true) @@ -271,7 +271,7 @@ public override void OnInspectorGUI() EditorGUILayout.PropertyField(_positionMode); // Auto Scaling - DrawAutoScaling(_autoScaling, targets.OfType()); + DrawAutoScaling(_autoScalingMode, targets.OfType()); // Target ParticleSystems. EditorGUI.BeginChangeCheck(); @@ -478,16 +478,17 @@ private static bool DrawMeshSharing(SerializedProperty spMeshSharing, Serialized private static void DrawAutoScaling(SerializedProperty prop, IEnumerable uiParticles) { + var isTransformMode = prop.intValue == (int)UIParticle.AutoScalingMode.Transform; EditorGUI.BeginChangeCheck(); EditorGUILayout.PropertyField(prop); - if (!EditorGUI.EndChangeCheck()) return; + if (!EditorGUI.EndChangeCheck() || !isTransformMode) return; // on changed true->false, reset scale. EditorApplication.delayCall += () => { foreach (var uip in uiParticles) { - if (!uip || uip.autoScaling) continue; + if (!uip) continue; uip.transform.localScale = Vector3.one; } }; @@ -509,7 +510,7 @@ private static void WindowFunction(Object target, SceneView sceneView) EditorGUILayout.PropertyField(s_SerializedObject.FindProperty("m_Enabled")); s_XYZMode = DrawFloatOrVector3Field(s_SerializedObject.FindProperty("m_Scale3D"), s_XYZMode); EditorGUILayout.PropertyField(s_SerializedObject.FindProperty("m_PositionMode")); - DrawAutoScaling(s_SerializedObject.FindProperty("m_AutoScaling"), uiParticles); + DrawAutoScaling(s_SerializedObject.FindProperty("m_AutoScalingMode"), uiParticles); EditorGUIUtility.labelWidth = labelWidth; } diff --git a/Scripts/UIParticle.cs b/Scripts/UIParticle.cs index 80e9b57..76599bb 100644 --- a/Scripts/UIParticle.cs +++ b/Scripts/UIParticle.cs @@ -1,10 +1,13 @@ +using System; using System.Collections.Generic; using System.Runtime.CompilerServices; using Coffee.UIParticleExtensions; +using UnityEditor; using UnityEngine; using UnityEngine.Rendering; using UnityEngine.Serialization; using UnityEngine.UI; +using Random = UnityEngine.Random; [assembly: InternalsVisibleTo("Coffee.UIParticle.Editor")] @@ -33,6 +36,13 @@ public enum PositionMode Absolute } + public enum AutoScalingMode + { + None, + UIParticle, + Transform + } + [HideInInspector] [SerializeField] internal bool m_IsTrail; @@ -82,10 +92,14 @@ public enum PositionMode private PositionMode m_PositionMode = PositionMode.Relative; [SerializeField] - [Tooltip("Transform.lossyScale (=world scale) is automatically set to (1, 1, 1), " + - "to prevent the root-Canvas scale from affecting the hierarchy-scaled ParticleSystem.")] + [Tooltip("Prevent the root-Canvas scale from affecting the hierarchy-scaled ParticleSystem.")] private bool m_AutoScaling = true; + [SerializeField] + [Tooltip("Transform: Transform.lossyScale (=world scale) will be set to (1, 1, 1)." + + "UIParticle: UIParticle.scale will be adjusted.")] + private AutoScalingMode m_AutoScalingMode = AutoScalingMode.Transform; + private readonly List _renderers = new List(); private int _groupId; private Camera _orthoCamera; @@ -167,16 +181,30 @@ public bool absoluteMode } /// - /// Transform.lossyScale (=world scale) will be set to (1, 1, 1) on update. - /// It prevents the root-Canvas scale from affecting the hierarchy-scaled ParticleSystem. + /// Prevents the root-Canvas scale from affecting the hierarchy-scaled ParticleSystem. /// + [Obsolete("The autoScaling is now obsolete. Please use the autoScalingMode instead.", false)] public bool autoScaling { - get { return m_AutoScaling; } + get { return m_AutoScalingMode != AutoScalingMode.None; } set { - if (m_AutoScaling == value) return; - m_AutoScaling = value; + autoScalingMode = value ? AutoScalingMode.Transform : AutoScalingMode.None; + } + } + + /// + /// Auto scaling mode. + /// Transform: Transform.lossyScale (=world scale) will be set to (1, 1, 1). + /// UIParticle: UIParticle.scale will be adjusted. + /// + public AutoScalingMode autoScalingMode + { + get { return m_AutoScalingMode; } + set + { + if (m_AutoScalingMode == value) return; + m_AutoScalingMode = value; UpdateTracker(); } } @@ -235,6 +263,14 @@ public Vector3 scale3D set { m_Scale3D = value; } } + /// + /// Particle effect scale. + /// + public Vector3 scale3DForCalc + { + get { return autoScalingMode == AutoScalingMode.UIParticle ? m_Scale3D.GetScaled(canvasScale) : m_Scale3D; } + } + public List particles { get { return m_Particles; } @@ -268,6 +304,8 @@ public override Material materialForRendering public Vector3 parentScale { get; private set; } + public Vector3 canvasScale { get; private set; } + protected override void OnEnable() { ResetGroupId(); @@ -328,7 +366,7 @@ void ISerializationCallbackReceiver.OnBeforeSerialize() void ISerializationCallbackReceiver.OnAfterDeserialize() { - if (m_IgnoreCanvasScaler) + if (m_IgnoreCanvasScaler || m_AutoScaling) { m_IgnoreCanvasScaler = false; m_AutoScaling = false; @@ -488,8 +526,9 @@ public void RefreshParticles(List particles) internal void UpdateTransformScale() { + canvasScale = canvas.rootCanvas.transform.localScale.Inverse(); parentScale = transform.parent.lossyScale; - if (!autoScaling) return; + if (autoScalingMode != AutoScalingMode.Transform) return; var newScale = parentScale.Inverse(); if (transform.localScale != newScale) @@ -623,7 +662,7 @@ private Camera GetBakeCamera() private void UpdateTracker() { - if (!enabled || !autoScaling) + if (!enabled || !autoScaling || autoScalingMode != AutoScalingMode.Transform) { _tracker.Clear(); } diff --git a/Scripts/UIParticleAttractor.cs b/Scripts/UIParticleAttractor.cs index 76460f0..a33c6a3 100644 --- a/Scripts/UIParticleAttractor.cs +++ b/Scripts/UIParticleAttractor.cs @@ -182,21 +182,22 @@ private Vector3 GetDestinationPosition() if (isUI) { var inverseScale = _uiParticle.parentScale.Inverse(); - dstPos = dstPos.GetScaled(inverseScale, _uiParticle.scale3D.Inverse()); + var scale3d = _uiParticle.scale3DForCalc; + dstPos = dstPos.GetScaled(inverseScale, scale3d.Inverse()); // Relative mode if (_uiParticle.positionMode == UIParticle.PositionMode.Relative) { var diff = _uiParticle.transform.position - psPos; - diff.Scale(_uiParticle.scale3D - inverseScale); - diff.Scale(_uiParticle.scale3D.Inverse()); + diff.Scale(scale3d - inverseScale); + diff.Scale(scale3d.Inverse()); dstPos += diff; } #if UNITY_EDITOR if (!Application.isPlaying && !isLocalSpace) { - dstPos += psPos - psPos.GetScaled(inverseScale, _uiParticle.scale3D.Inverse()); + dstPos += psPos - psPos.GetScaled(inverseScale, scale3d.Inverse()); } #endif } diff --git a/Scripts/UIParticleRenderer.cs b/Scripts/UIParticleRenderer.cs index 44bf3d8..c5406d7 100644 --- a/Scripts/UIParticleRenderer.cs +++ b/Scripts/UIParticleRenderer.cs @@ -260,7 +260,7 @@ public void UpdateMesh(Camera bakeCamera) !isActiveAndEnabled || !_particleSystem || !_parent || !canvasRenderer || !canvas || !bakeCamera || _parent.meshSharing == UIParticle.MeshSharing.Replica - || !transform.lossyScale.GetScaled(_parent.scale3D).IsVisible() // Scale is not visible. + || !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 @@ -462,7 +462,7 @@ public override void Cull(Rect clipRect, bool validRect) private Vector3 GetWorldScale() { Profiler.BeginSample("[UIParticleRenderer] GetWorldScale"); - var scale = _parent.scale3D.GetScaled(_parent.parentScale); + var scale = _parent.scale3DForCalc.GetScaled(_parent.parentScale); Profiler.EndSample(); return scale; }