using UnityEngine; using UnityEngine.UI; namespace Coffee.UIExtensions { /// /// Transition effect. /// [AddComponentMenu("UI/UIEffect/UITransitionEffect", 5)] public class UITransitionEffect : UIEffectBase { //################################ // Constant or Static Members. //################################ public const string shaderName = "UI/Hidden/UI-Effect-Transition"; static readonly ParameterTexture _ptex = new ParameterTexture(8, 128, "_ParamTex"); /// /// Effect mode. /// public enum EffectMode { Fade = 1, Cutoff = 2, Dissolve = 3, } //################################ // Serialize Members. //################################ [Tooltip("Effect mode.")] [SerializeField] EffectMode m_EffectMode = EffectMode.Cutoff; [Tooltip("Effect factor between 0(hidden) and 1(shown).")] [SerializeField][Range(0, 1)] float m_EffectFactor = 1; [Tooltip("Transition texture (single channel texture).")] [SerializeField] Texture m_TransitionTexture; [Header("Advanced Option")] [Tooltip("The area for effect.")] [SerializeField] EffectArea m_EffectArea = EffectArea.RectTransform; [Tooltip("Keep effect aspect ratio.")] [SerializeField] bool m_KeepAspectRatio; [Tooltip("Dissolve edge width.")] [SerializeField][Range(0, 1)] float m_DissolveWidth = 0.5f; [Tooltip("Dissolve edge softness.")] [SerializeField][Range(0, 1)] float m_DissolveSoftness = 0.5f; [Tooltip("Dissolve edge color.")] [SerializeField][ColorUsage(false)] Color m_DissolveColor = new Color(0.0f, 0.25f, 1.0f); [Tooltip("Disable graphic's raycast target on hidden.")] [SerializeField] bool m_PassRayOnHidden; [Header("Effect Player")] [SerializeField] EffectPlayer m_Player; //################################ // Public Members. //################################ /// /// Effect factor between 0(no effect) and 1(complete effect). /// public float effectFactor { get { return m_EffectFactor; } set { value = Mathf.Clamp(value, 0, 1); if (!Mathf.Approximately(m_EffectFactor, value)) { m_EffectFactor = value; SetDirty(); } } } /// /// Transition texture. /// public Texture transitionTexture { get { return m_TransitionTexture; } set { if (m_TransitionTexture != value) { m_TransitionTexture = value; if (graphic) { ModifyMaterial(); } } } } /// /// Effect mode. /// public EffectMode effectMode { get { return m_EffectMode; } } /// /// Keep aspect ratio. /// public bool keepAspectRatio { get { return m_KeepAspectRatio; } set { if (m_KeepAspectRatio != value) { m_KeepAspectRatio = value; targetGraphic.SetVerticesDirty(); } } } /// /// Gets the parameter texture. /// public override ParameterTexture ptex { get { return _ptex; } } /// /// Dissolve edge width. /// public float dissolveWidth { get { return m_DissolveWidth; } set { value = Mathf.Clamp(value, 0, 1); if (!Mathf.Approximately(m_DissolveWidth, value)) { m_DissolveWidth = value; SetDirty(); } } } /// /// Dissolve edge softness. /// public float dissolveSoftness { get { return m_DissolveSoftness; } set { value = Mathf.Clamp(value, 0, 1); if (!Mathf.Approximately(m_DissolveSoftness, value)) { m_DissolveSoftness = value; SetDirty(); } } } /// /// Dissolve edge color. /// public Color dissolveColor { get { return m_DissolveColor; } set { if (m_DissolveColor != value) { m_DissolveColor = value; SetDirty(); } } } /// /// Duration for showing/hiding. /// public float duration { get { return _player.duration; } set { _player.duration = Mathf.Max(value, 0.1f); } } /// /// Disable graphic's raycast target on hidden. /// public bool passRayOnHidden { get { return m_PassRayOnHidden; } set { m_PassRayOnHidden = value; } } /// /// Update mode for showing/hiding. /// public AnimatorUpdateMode updateMode { get { return _player.updateMode; } set { _player.updateMode = value; } } /// /// Show transition. /// public void Show(bool reset = true) { _player.loop = false; _player.Play(reset, f => effectFactor = f); } /// /// Hide transition. /// public void Hide(bool reset = true) { _player.loop = false; _player.Play(reset, f => effectFactor = 1 - f); } /// /// Modifies the material. /// public override void ModifyMaterial() { if (isTMPro) { return; } ulong hash = (m_TransitionTexture ? (uint)m_TransitionTexture.GetInstanceID() : 0) + ((ulong)2 << 32) + ((ulong)m_EffectMode << 36); if (_materialCache != null && (_materialCache.hash != hash || !isActiveAndEnabled || !m_EffectMaterial)) { MaterialCache.Unregister(_materialCache); _materialCache = null; } if (!isActiveAndEnabled || !m_EffectMaterial) { material = null; } else if (!m_TransitionTexture) { material = m_EffectMaterial; } else if (_materialCache != null && _materialCache.hash == hash) { material = _materialCache.material; } else { _materialCache = MaterialCache.Register(hash, m_TransitionTexture, () => { var mat = new Material(m_EffectMaterial); mat.name += "_" + m_TransitionTexture.name; mat.SetTexture("_NoiseTex", m_TransitionTexture); return mat; }); material = _materialCache.material; } } /// /// Modifies the mesh. /// public override void ModifyMesh(VertexHelper vh) { if (!isActiveAndEnabled) { return; } bool isText = isTMPro || graphic is Text; float normalizedIndex = ptex.GetNormalizedIndex(this); // rect. var tex = transitionTexture; var aspectRatio = m_KeepAspectRatio && tex ? ((float)tex.width) / tex.height : -1; Rect rect = m_EffectArea.GetEffectArea(vh, rectTransform.rect, aspectRatio); // Set prameters to vertex. UIVertex vertex = default(UIVertex); float x, y; int count = vh.currentVertCount; for (int i = 0; i < count; i++) { vh.PopulateUIVertex(ref vertex, i); m_EffectArea.GetPositionFactor(i, rect, vertex.position, isText, isTMPro, out x, out y); vertex.uv0 = new Vector2( Packer.ToFloat(vertex.uv0.x, vertex.uv0.y), Packer.ToFloat(x, y, normalizedIndex) ); vh.SetUIVertex(vertex, i); } } //################################ // Protected Members. //################################ /// /// This function is called when the object becomes enabled and active. /// protected override void OnEnable() { base.OnEnable(); _player.OnEnable(null); _player.loop = false; } /// /// This function is called when the behaviour becomes disabled () or inactive. /// protected override void OnDisable() { base.OnDisable(); MaterialCache.Unregister(_materialCache); _materialCache = null; _player.OnDisable(); } protected override void SetDirty() { foreach (var m in materials) { ptex.RegisterMaterial(m); } ptex.SetData(this, 0, m_EffectFactor); // param1.x : effect factor if (m_EffectMode == EffectMode.Dissolve) { ptex.SetData(this, 1, m_DissolveWidth); // param1.y : width ptex.SetData(this, 2, m_DissolveSoftness); // param1.z : softness ptex.SetData(this, 4, m_DissolveColor.r); // param2.x : red ptex.SetData(this, 5, m_DissolveColor.g); // param2.y : green ptex.SetData(this, 6, m_DissolveColor.b); // param2.z : blue } // Disable graphic's raycastTarget on hidden. if (m_PassRayOnHidden) { targetGraphic.raycastTarget = 0 < m_EffectFactor; } } #if UNITY_EDITOR /// /// Gets the material. /// /// The material. protected override Material GetMaterial() { return MaterialResolver.GetOrGenerateMaterialVariant(Shader.Find(shaderName), m_EffectMode); } #endif //################################ // Private Members. //################################ MaterialCache _materialCache = null; EffectPlayer _player { get { return m_Player ?? (m_Player = new EffectPlayer()); } } } }