forked from cgvr/DeltaVR
		
	
		
			
				
	
	
		
			363 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			363 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
using System.Collections.Generic;
 | 
						|
using UnityEngine.XR.Interaction.Toolkit;
 | 
						|
 | 
						|
namespace UnityEngine.XR.Content.Rendering
 | 
						|
{
 | 
						|
    /// <summary>
 | 
						|
    /// All-in-one controller for animated object highlights in different states - hovered, selected, and activated
 | 
						|
    /// </summary>
 | 
						|
    public class InteractableVisualsController : MonoBehaviour
 | 
						|
    {
 | 
						|
        const float k_ShineTime = 0.2f;
 | 
						|
 | 
						|
        enum PriorityHighlightingState
 | 
						|
        {
 | 
						|
            Unknown,
 | 
						|
            Highlighted,
 | 
						|
            Unhighlighted,
 | 
						|
        }
 | 
						|
 | 
						|
        static List<IXRTargetPriorityInteractor> s_InteractorList = new List<IXRTargetPriorityInteractor>();
 | 
						|
 | 
						|
#pragma warning disable 649
 | 
						|
        [Header("Audio")]
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("The hover audio source.")]
 | 
						|
        AudioSource m_AudioHover;
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("The click audio source.")]
 | 
						|
        AudioSource m_AudioClick;
 | 
						|
 | 
						|
        [Header("Visual")]
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("Material capture settings.")]
 | 
						|
        HighlightController m_HighlightController = new HighlightController();
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("The outline highlight for selection.")]
 | 
						|
        OutlineHighlight m_OutlineHighlight;
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("The material highlight for hover.")]
 | 
						|
        MaterialHighlight m_MaterialHighlight;
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("The outline hover color.")]
 | 
						|
        Color m_HoverColor = new Color(0.25f, 0.7f, 0.9f, 1f);
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("The outline hover color when the Interactable has the highest priority for selection.")]
 | 
						|
        Color m_HoverPriorityColor = new Color(0.09411765f, 0.4392157f, 0.7137255f, 1f);
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("The outline selection color.")]
 | 
						|
        Color m_SelectionColor = new Color(1f, 0.4f, 0f, 1f);
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("To play material activate anim.")]
 | 
						|
        bool m_PlayMaterialActivateAnim;
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("To play outline activate anim.")]
 | 
						|
        bool m_PlayOutlineActivateAnim;
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("If true, the highlight state will be on during hover.")]
 | 
						|
        bool m_HighlightOnHover = true;
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("If true, the highlight state will be on during hover when the Interactable has the highest priority for selection.")]
 | 
						|
        bool m_HighlightOnHoverPriority = true;
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("If true, the highlight state will be on during select.")]
 | 
						|
        bool m_HighlightOnSelect = true;
 | 
						|
 | 
						|
        [SerializeField]
 | 
						|
        [Tooltip("If true, the highlight state will be on during activate.")]
 | 
						|
        bool m_HighlightOnActivate = true;
 | 
						|
 | 
						|
#pragma warning restore 649
 | 
						|
 | 
						|
        XRBaseInteractable m_Interactable;
 | 
						|
 | 
						|
        Material m_PulseMaterial;
 | 
						|
        float m_StartingAlpha;
 | 
						|
        float m_StartingWidth;
 | 
						|
 | 
						|
        int m_SelectedCount;
 | 
						|
        int m_HoveredCount;
 | 
						|
        bool m_Highlighting;
 | 
						|
        PriorityHighlightingState m_PriorityHighlightingState;
 | 
						|
 | 
						|
        bool m_PlayShine;
 | 
						|
        float m_ShineTimer;
 | 
						|
 | 
						|
        bool isActivated { get; set; }
 | 
						|
        bool isSelected => m_SelectedCount > 0;
 | 
						|
        bool isHovered => m_HoveredCount > 0;
 | 
						|
 | 
						|
        /// <summary>
 | 
						|
        /// See <see cref="MonoBehaviour"/>.
 | 
						|
        /// </summary>
 | 
						|
        protected void Awake()
 | 
						|
        {
 | 
						|
            // Find the grab interactable
 | 
						|
            m_Interactable = GetComponentInParent<XRBaseInteractable>();
 | 
						|
 | 
						|
            // Hook up to events
 | 
						|
            if (m_Interactable is IXRHoverInteractable hoverInteractable)
 | 
						|
            {
 | 
						|
                hoverInteractable.hoverEntered.AddListener(OnHoverEntered);
 | 
						|
                hoverInteractable.hoverExited.AddListener(OnHoverExited);
 | 
						|
            }
 | 
						|
 | 
						|
            if (m_Interactable is IXRSelectInteractable selectInteractable)
 | 
						|
            {
 | 
						|
                selectInteractable.selectEntered.AddListener(OnSelectEntered);
 | 
						|
                selectInteractable.selectExited.AddListener(OnSelectExited);
 | 
						|
            }
 | 
						|
 | 
						|
            if (m_Interactable is IXRActivateInteractable activateInteractable)
 | 
						|
            {
 | 
						|
                activateInteractable.activated.AddListener(OnActivated);
 | 
						|
                activateInteractable.deactivated.AddListener(OnDeactivated);
 | 
						|
            }
 | 
						|
 | 
						|
            // Cache materials for highlighting
 | 
						|
            m_HighlightController.rendererSource = m_Interactable.transform;
 | 
						|
 | 
						|
            // Tell the highlight objects to get renderers starting at the grab interactable down
 | 
						|
            if (m_MaterialHighlight != null)
 | 
						|
            {
 | 
						|
                m_HighlightController.RegisterCacheUser(m_MaterialHighlight);
 | 
						|
                m_PulseMaterial = m_MaterialHighlight.highlightMaterial;
 | 
						|
 | 
						|
                if (m_PulseMaterial != null)
 | 
						|
                    m_StartingAlpha = m_PulseMaterial.GetFloat("_PulseMinAlpha");
 | 
						|
            }
 | 
						|
            if (m_OutlineHighlight != null)
 | 
						|
                m_HighlightController.RegisterCacheUser(m_OutlineHighlight);
 | 
						|
 | 
						|
            m_HighlightController.Initialize();
 | 
						|
            m_StartingWidth = m_OutlineHighlight.outlineScale;
 | 
						|
        }
 | 
						|
 | 
						|
        /// <summary>
 | 
						|
        /// See <see cref="MonoBehaviour"/>.
 | 
						|
        /// </summary>
 | 
						|
        protected void Update()
 | 
						|
        {
 | 
						|
            UpdatePriorityHighlightingState();
 | 
						|
            m_HighlightController.Update();
 | 
						|
            if (m_MaterialHighlight != null)
 | 
						|
            {
 | 
						|
                // Do timer count up/count down
 | 
						|
                if (m_PlayShine)
 | 
						|
                {
 | 
						|
                    m_ShineTimer += Time.deltaTime;
 | 
						|
 | 
						|
                    var shinePercent = Mathf.Clamp01(m_ShineTimer / k_ShineTime);
 | 
						|
                    var shineValue = Mathf.PingPong(shinePercent, 0.5f) * 2.0f;
 | 
						|
 | 
						|
                    m_PulseMaterial.SetFloat("_PulseMinAlpha", Mathf.Lerp(m_StartingAlpha, 1f, shineValue));
 | 
						|
 | 
						|
                    if (shinePercent >= 1.0f)
 | 
						|
                    {
 | 
						|
                        m_PlayShine = false;
 | 
						|
                        m_ShineTimer = 0.0f;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        void UpdateHighlightState()
 | 
						|
        {
 | 
						|
            var shouldHighlight = false;
 | 
						|
 | 
						|
            if (isActivated)
 | 
						|
                shouldHighlight = m_HighlightOnActivate;
 | 
						|
            else
 | 
						|
            {
 | 
						|
                if (isSelected)
 | 
						|
                    shouldHighlight = m_HighlightOnSelect;
 | 
						|
                else if (isHovered)
 | 
						|
                    shouldHighlight = m_HighlightOnHover || (m_HighlightOnHoverPriority && m_PriorityHighlightingState == PriorityHighlightingState.Highlighted);
 | 
						|
            }
 | 
						|
 | 
						|
            if (shouldHighlight == m_Highlighting)
 | 
						|
                return;
 | 
						|
 | 
						|
            m_Highlighting = shouldHighlight;
 | 
						|
 | 
						|
            if (m_Highlighting)
 | 
						|
                m_HighlightController.Highlight();
 | 
						|
            else
 | 
						|
                m_HighlightController.Unhighlight();
 | 
						|
        }
 | 
						|
 | 
						|
        void OnHoverEntered(HoverEnterEventArgs args)
 | 
						|
        {
 | 
						|
            if (args.interactorObject is XRSocketInteractor)
 | 
						|
                return;
 | 
						|
 | 
						|
            m_HoveredCount++;
 | 
						|
 | 
						|
            if (isSelected)
 | 
						|
                return;
 | 
						|
 | 
						|
            if (m_AudioHover != null)
 | 
						|
                m_AudioHover.Play();
 | 
						|
 | 
						|
            if (m_MaterialHighlight != null)
 | 
						|
                m_PulseMaterial.color = m_HoverColor;
 | 
						|
 | 
						|
            if (m_OutlineHighlight != null)
 | 
						|
                m_OutlineHighlight.outlineColor = m_HoverColor;
 | 
						|
 | 
						|
            m_PriorityHighlightingState = PriorityHighlightingState.Unknown;
 | 
						|
            UpdateHighlightState();
 | 
						|
        }
 | 
						|
 | 
						|
        void OnHoverExited(HoverExitEventArgs args)
 | 
						|
        {
 | 
						|
            if (args.interactorObject is XRSocketInteractor)
 | 
						|
                return;
 | 
						|
 | 
						|
            m_HoveredCount--;
 | 
						|
            m_PriorityHighlightingState = PriorityHighlightingState.Unknown;
 | 
						|
            UpdateHighlightState();
 | 
						|
        }
 | 
						|
 | 
						|
        bool HasValidInteractor(List<IXRTargetPriorityInteractor> interactors)
 | 
						|
        {
 | 
						|
            foreach (var interactor in interactors)
 | 
						|
            {
 | 
						|
                if (!(interactor is XRSocketInteractor))
 | 
						|
                    return true;
 | 
						|
            }
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
 | 
						|
        void UpdatePriorityHighlightingState()
 | 
						|
        {
 | 
						|
            if (!m_HighlightOnHoverPriority || !isHovered || isSelected)
 | 
						|
                return;
 | 
						|
 | 
						|
            var manager = m_Interactable.interactionManager;
 | 
						|
            if (manager == null)
 | 
						|
                return;
 | 
						|
 | 
						|
            var highestPriorityForSelection = manager.IsHighestPriorityTarget(m_Interactable, s_InteractorList);
 | 
						|
            if (!HasValidInteractor(s_InteractorList))
 | 
						|
                return;
 | 
						|
 | 
						|
            if (highestPriorityForSelection && m_PriorityHighlightingState != PriorityHighlightingState.Highlighted)
 | 
						|
            {
 | 
						|
                m_PriorityHighlightingState = PriorityHighlightingState.Highlighted;
 | 
						|
 | 
						|
                if (m_PulseMaterial != null)
 | 
						|
                    m_PulseMaterial.color = m_HoverPriorityColor;
 | 
						|
 | 
						|
                if (m_OutlineHighlight != null)
 | 
						|
                    m_OutlineHighlight.outlineColor = m_HoverPriorityColor;
 | 
						|
 | 
						|
                UpdateHighlightState();
 | 
						|
            }
 | 
						|
 | 
						|
            if (!highestPriorityForSelection && m_PriorityHighlightingState != PriorityHighlightingState.Unhighlighted)
 | 
						|
            {
 | 
						|
                m_PriorityHighlightingState = PriorityHighlightingState.Unhighlighted;
 | 
						|
 | 
						|
                if (m_PulseMaterial != null)
 | 
						|
                    m_PulseMaterial.color = m_HoverColor;
 | 
						|
 | 
						|
                if (m_OutlineHighlight != null)
 | 
						|
                    m_OutlineHighlight.outlineColor = m_HoverColor;
 | 
						|
 | 
						|
                UpdateHighlightState();
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        void OnSelectEntered(SelectEnterEventArgs args)
 | 
						|
        {
 | 
						|
            if (args.interactorObject is XRSocketInteractor)
 | 
						|
                return;
 | 
						|
 | 
						|
            if (m_AudioClick != null)
 | 
						|
                m_AudioClick.Play();
 | 
						|
 | 
						|
            if (m_OutlineHighlight != null)
 | 
						|
            {
 | 
						|
                m_OutlineHighlight.outlineColor = m_SelectionColor;
 | 
						|
                m_OutlineHighlight.PlayPulseAnimation();
 | 
						|
            }
 | 
						|
 | 
						|
            if (m_MaterialHighlight != null)
 | 
						|
                m_PulseMaterial.color = m_SelectionColor;
 | 
						|
 | 
						|
            m_SelectedCount++;
 | 
						|
            UpdateHighlightState();
 | 
						|
        }
 | 
						|
 | 
						|
        void OnSelectExited(SelectExitEventArgs args)
 | 
						|
        {
 | 
						|
            if (args.interactorObject is XRSocketInteractor)
 | 
						|
                return;
 | 
						|
 | 
						|
            if (m_OutlineHighlight != null)
 | 
						|
                m_OutlineHighlight.outlineColor = m_HoverColor;
 | 
						|
            if (m_MaterialHighlight != null)
 | 
						|
                m_PulseMaterial.color = m_HoverColor;
 | 
						|
 | 
						|
            m_OutlineHighlight.PlayPulseAnimation();
 | 
						|
 | 
						|
            // In case the Interactable is dropped while activated.
 | 
						|
            isActivated = false;
 | 
						|
            m_SelectedCount--;
 | 
						|
            m_PriorityHighlightingState = PriorityHighlightingState.Unknown;
 | 
						|
            UpdateHighlightState();
 | 
						|
        }
 | 
						|
 | 
						|
        void OnActivated(ActivateEventArgs args)
 | 
						|
        {
 | 
						|
            if (args.interactorObject is XRSocketInteractor)
 | 
						|
                return;
 | 
						|
 | 
						|
            if (m_OutlineHighlight != null)
 | 
						|
            {
 | 
						|
                if (m_PlayMaterialActivateAnim)
 | 
						|
                    m_PlayShine = true;
 | 
						|
 | 
						|
                if (m_PlayOutlineActivateAnim)
 | 
						|
                {
 | 
						|
                    m_OutlineHighlight.outlineScale = 1f;
 | 
						|
                    m_OutlineHighlight.PlayPulseAnimation();
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            isActivated = true;
 | 
						|
            UpdateHighlightState();
 | 
						|
        }
 | 
						|
 | 
						|
        void OnDeactivated(DeactivateEventArgs args)
 | 
						|
        {
 | 
						|
            if (args.interactorObject is XRSocketInteractor)
 | 
						|
                return;
 | 
						|
 | 
						|
            if (m_OutlineHighlight != null)
 | 
						|
            {
 | 
						|
                if (m_PlayOutlineActivateAnim)
 | 
						|
                {
 | 
						|
                    m_OutlineHighlight.outlineScale = m_StartingWidth;
 | 
						|
                    m_OutlineHighlight.PlayPulseAnimation();
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            isActivated = false;
 | 
						|
            UpdateHighlightState();
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |