409 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C#
		
	
	
	
			
		
		
	
	
			409 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C#
		
	
	
	
| using System;
 | |
| using System.Collections;
 | |
| using System.Collections.Generic;
 | |
| using UnityEngine;
 | |
| using UnityEngine.EventSystems;
 | |
| 
 | |
| namespace HighlightPlus {
 | |
| 
 | |
|     public enum TriggerMode {
 | |
|         ColliderEventsOnlyOnThisObject = 0,
 | |
|         RaycastOnThisObjectAndChildren = 1,
 | |
|         Volume = 2
 | |
|     }
 | |
| 
 | |
|     public enum RayCastSource {
 | |
|         MousePosition = 0,
 | |
|         CameraDirection = 1
 | |
|     }
 | |
| 
 | |
| 
 | |
|     [RequireComponent(typeof(HighlightEffect))]
 | |
|     [ExecuteInEditMode]
 | |
|     [HelpURL("https://kronnect.com/guides/highlight-plus-introduction/")]
 | |
|     public class HighlightTrigger : MonoBehaviour {
 | |
| 
 | |
|         [Tooltip("Enables highlight when pointer is over this object.")]
 | |
|         public bool highlightOnHover = true;
 | |
|         [Tooltip("Used to trigger automatic highlighting including children objects.")]
 | |
| #if ENABLE_INPUT_SYSTEM
 | |
|         public TriggerMode triggerMode = TriggerMode.RaycastOnThisObjectAndChildren;
 | |
| #else
 | |
|         public TriggerMode triggerMode = TriggerMode.ColliderEventsOnlyOnThisObject;
 | |
| #endif
 | |
|         public Camera raycastCamera;
 | |
|         public RayCastSource raycastSource = RayCastSource.MousePosition;
 | |
|         public LayerMask raycastLayerMask = -1;
 | |
|         [Tooltip("Minimum distance for target.")]
 | |
|         public float minDistance;
 | |
|         [Tooltip("Maximum distance for target. 0 = infinity")]
 | |
|         public float maxDistance;
 | |
|         [Tooltip("Blocks interaction if pointer is over an UI element")]
 | |
|         public bool respectUI = true;
 | |
|         public LayerMask volumeLayerMask;
 | |
| 
 | |
|         const int MAX_RAYCAST_HITS = 100;
 | |
| 
 | |
| 
 | |
|         [Tooltip("If the object will be selected by clicking with mouse or tapping on it.")]
 | |
|         public bool selectOnClick;
 | |
|         [Tooltip("Profile to use when object is selected by clicking on it.")]
 | |
|         public HighlightProfile selectedProfile;
 | |
|         [Tooltip("Profile to use whtn object is selected and highlighted.")]
 | |
|         public HighlightProfile selectedAndHighlightedProfile;
 | |
|         [Tooltip("Automatically deselects any other selected object prior selecting this one")]
 | |
|         public bool singleSelection;
 | |
|         [Tooltip("Toggles selection on/off when clicking object")]
 | |
|         public bool toggle;
 | |
|         [Tooltip("Keeps current selection when clicking outside of any selectable object")]
 | |
|         public bool keepSelection = true;
 | |
| 
 | |
|         [NonSerialized] public Collider[] colliders;
 | |
| 
 | |
|         Collider currentCollider;
 | |
|         static RaycastHit[] hits;
 | |
|         HighlightEffect hb;
 | |
| 
 | |
|         public HighlightEffect highlightEffect { get { return hb; } }
 | |
| 
 | |
|         public event OnObjectSelectionEvent OnObjectSelected;
 | |
|         public event OnObjectSelectionEvent OnObjectUnSelected;
 | |
|         public event OnObjectHighlightEvent OnObjectHighlightStart;
 | |
|         public event OnObjectHighlightEvent OnObjectHighlightEnd;
 | |
| 
 | |
|         TriggerMode currentTriggerMode;
 | |
| 
 | |
|         [RuntimeInitializeOnLoadMethod]
 | |
|         void DomainReloadDisabledSupport() {
 | |
|             HighlightManager.selectedObjects.Clear();
 | |
|         }
 | |
| 
 | |
|         void OnEnable() {
 | |
|             Init();
 | |
|         }
 | |
| 
 | |
|         private void OnValidate() {
 | |
|             if (currentTriggerMode != triggerMode) {
 | |
|                 currentTriggerMode = triggerMode;
 | |
|                 if (currentTriggerMode == TriggerMode.RaycastOnThisObjectAndChildren) {
 | |
|                     colliders = GetComponentsInChildren<Collider>();
 | |
|                     if (hits == null || hits.Length != MAX_RAYCAST_HITS) {
 | |
|                         hits = new RaycastHit[MAX_RAYCAST_HITS];
 | |
|                     }
 | |
|                     if (Application.isPlaying) {
 | |
|                         StopAllCoroutines();
 | |
|                         if (gameObject.activeInHierarchy) {
 | |
|                             StartCoroutine(DoRayCast());
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void Init() {
 | |
|             if (raycastCamera == null) {
 | |
|                 raycastCamera = HighlightManager.GetCamera();
 | |
|             }
 | |
|             currentTriggerMode = triggerMode;
 | |
|             if (triggerMode == TriggerMode.RaycastOnThisObjectAndChildren) {
 | |
|                 colliders = GetComponentsInChildren<Collider>();
 | |
|             }
 | |
|             if (hb == null) {
 | |
|                 hb = GetComponent<HighlightEffect>();
 | |
|             }
 | |
|             InputProxy.Init();
 | |
|         }
 | |
| 
 | |
|         void Start() {
 | |
|             if (triggerMode == TriggerMode.RaycastOnThisObjectAndChildren) {
 | |
|                 if (raycastCamera == null) {
 | |
|                     raycastCamera = HighlightManager.GetCamera();
 | |
|                     if (raycastCamera == null) {
 | |
|                         Debug.LogError("Highlight Trigger on " + gameObject.name + ": no camera found!");
 | |
|                     }
 | |
|                 }
 | |
|                 if (colliders != null && colliders.Length > 0) {
 | |
|                     hits = new RaycastHit[MAX_RAYCAST_HITS];
 | |
|                     if (Application.isPlaying) {
 | |
|                         StopAllCoroutines();
 | |
|                         StartCoroutine(DoRayCast());
 | |
|                     }
 | |
|                 }
 | |
|             } else {
 | |
|                 Collider collider = GetComponent<Collider>();
 | |
|                 if (collider == null) {
 | |
|                     if (GetComponent<MeshFilter>() != null) {
 | |
|                         gameObject.AddComponent<MeshCollider>();
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         IEnumerator DoRayCast() {
 | |
|             yield return null;
 | |
|             WaitForEndOfFrame w = new WaitForEndOfFrame();
 | |
|             while (triggerMode == TriggerMode.RaycastOnThisObjectAndChildren) {
 | |
|                 if (raycastCamera == null) {
 | |
|                     yield return null;
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 int hitCount;
 | |
|                 bool hit = false;
 | |
| 
 | |
| #if ENABLE_INPUT_SYSTEM
 | |
| 
 | |
|                 if (respectUI) {
 | |
|                     EventSystem es = EventSystem.current;
 | |
|                     if (es == null) {
 | |
|                         es = CreateEventSystem();
 | |
|                     }
 | |
|                     List<RaycastResult> raycastResults = new List<RaycastResult>();
 | |
|                     PointerEventData eventData = new PointerEventData(es);
 | |
|                     Vector3 cameraPos = raycastCamera.transform.position;
 | |
|                     if (raycastSource == RayCastSource.MousePosition) {
 | |
|                         eventData.position = InputProxy.mousePosition;
 | |
|                     } else {
 | |
|                         eventData.position = new Vector2(raycastCamera.pixelWidth * 0.5f, raycastCamera.pixelHeight * 0.5f);
 | |
|                     }
 | |
|                     es.RaycastAll(eventData, raycastResults);
 | |
|                     hitCount = raycastResults.Count;
 | |
|                     // check UI blocker
 | |
|                     bool blocked = false;
 | |
|                     for (int k = 0; k < hitCount; k++) {
 | |
|                         RaycastResult rr = raycastResults[k];
 | |
|                         if (rr.module is UnityEngine.UI.GraphicRaycaster) {
 | |
|                             blocked = true;
 | |
|                             break;
 | |
|                         }
 | |
|                     }
 | |
|                     if (blocked) {
 | |
|                         yield return null;
 | |
|                         continue;
 | |
|                     }
 | |
|                     // look for our gameobject
 | |
|                     for (int k = 0; k < hitCount; k++) {
 | |
|                         RaycastResult rr = raycastResults[k];
 | |
|                         float distance = Vector3.Distance(rr.worldPosition, cameraPos);
 | |
|                         if (distance < minDistance || (maxDistance > 0 && distance > maxDistance)) continue;
 | |
| 
 | |
|                         GameObject theGameObject = rr.gameObject;
 | |
|                         for (int c = 0; c < colliders.Length; c++) {
 | |
|                             if (colliders[c].gameObject == theGameObject) {
 | |
|                                 Collider theCollider = colliders[c];
 | |
|                                 hit = true;
 | |
|                                 if (selectOnClick && InputProxy.GetMouseButtonDown(0)) {
 | |
|                                     ToggleSelection();
 | |
|                                     break;
 | |
|                                 } else if (theCollider != currentCollider) {
 | |
|                                     SwitchCollider(theCollider);
 | |
|                                     k = hitCount;
 | |
|                                     break;
 | |
|                                 }
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 // if not blocked by UI and no hit found, fallback to raycast (required if no PhysicsRaycaster is present on the camera)
 | |
| 
 | |
| #endif
 | |
|                 Ray ray;
 | |
|                 if (raycastSource == RayCastSource.MousePosition) {
 | |
| #if !ENABLE_INPUT_SYSTEM
 | |
| 
 | |
|                     if (!CanInteract()) {
 | |
|                         yield return null;
 | |
|                         continue;
 | |
|                     }
 | |
| #endif
 | |
|                     ray = raycastCamera.ScreenPointToRay(InputProxy.mousePosition);
 | |
|                 } else {
 | |
|                     ray = new Ray(raycastCamera.transform.position, raycastCamera.transform.forward);
 | |
|                 }
 | |
|                 if (maxDistance > 0) {
 | |
|                     hitCount = Physics.RaycastNonAlloc(ray, hits, maxDistance, raycastLayerMask);
 | |
|                 } else {
 | |
|                     hitCount = Physics.RaycastNonAlloc(ray, hits, float.MaxValue, raycastLayerMask);
 | |
|                 }
 | |
|                 bool isMouseButonDown = InputProxy.GetMouseButtonDown(0);
 | |
|                 for (int k = 0; k < hitCount; k++) {
 | |
|                     if (Vector3.Distance(hits[k].point, ray.origin) < minDistance) continue;
 | |
|                     Collider theCollider = hits[k].collider;
 | |
|                     for (int c = 0; c < colliders.Length; c++) {
 | |
|                         if (colliders[c] == theCollider) {
 | |
|                             hit = true;
 | |
|                             if (selectOnClick && isMouseButonDown) {
 | |
|                                 ToggleSelection();
 | |
|                                 break;
 | |
|                             } else if (theCollider != currentCollider) {
 | |
|                                 SwitchCollider(theCollider);
 | |
|                                 k = hitCount;
 | |
|                                 break;
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
| 
 | |
|                 if (!hit && currentCollider != null) {
 | |
|                     SwitchCollider(null);
 | |
|                 }
 | |
| 
 | |
|                 if (selectOnClick && isMouseButonDown && !keepSelection && !hit) {
 | |
|                     yield return w; // wait for other potential triggers to act
 | |
|                     if (HighlightManager.lastTriggerFrame < Time.frameCount) {
 | |
|                         HighlightManager.DeselectAll();
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 yield return null;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if ENABLE_INPUT_SYSTEM
 | |
|         EventSystem CreateEventSystem() {
 | |
|             GameObject eo = new GameObject("Event System created by Highlight Plus", typeof(EventSystem), typeof(UnityEngine.InputSystem.UI.InputSystemUIInputModule));
 | |
|             return eo.GetComponent<EventSystem>();
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|         void SwitchCollider(Collider newCollider) {
 | |
|             if (!highlightOnHover && !hb.isSelected) return;
 | |
| 
 | |
|             currentCollider = newCollider;
 | |
|             if (currentCollider != null) {
 | |
|                 Highlight(true);
 | |
|             } else {
 | |
|                 Highlight(false);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         bool CanInteract() {
 | |
|             if (!respectUI) return true;
 | |
|             EventSystem es = EventSystem.current;
 | |
|             if (es == null) return true;
 | |
|             if (Application.isMobilePlatform && InputProxy.touchCount > 0 && es.IsPointerOverGameObject(InputProxy.GetFingerIdFromTouch(0))) {
 | |
|                 return false;
 | |
|             } else if (es.IsPointerOverGameObject(-1))
 | |
|                 return false;
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
| 
 | |
|         void OnMouseDown() {
 | |
|             if (isActiveAndEnabled && triggerMode == TriggerMode.ColliderEventsOnlyOnThisObject) {
 | |
|                 if (!CanInteract()) return;
 | |
|                 if (selectOnClick && InputProxy.GetMouseButtonDown(0)) {
 | |
|                     ToggleSelection();
 | |
|                     return;
 | |
|                 }
 | |
|                 Highlight(true);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void OnMouseEnter() {
 | |
|             if (isActiveAndEnabled && triggerMode == TriggerMode.ColliderEventsOnlyOnThisObject) {
 | |
|                 if (!CanInteract()) return;
 | |
|                 Highlight(true);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void OnMouseExit() {
 | |
|             if (isActiveAndEnabled && triggerMode == TriggerMode.ColliderEventsOnlyOnThisObject) {
 | |
|                 if (!CanInteract()) return;
 | |
|                 Highlight(false);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void Highlight(bool state) {
 | |
|             if (state) {
 | |
|                 if (!hb.highlighted) {
 | |
|                     if (OnObjectHighlightStart != null && hb.target != null) {
 | |
|                         if (!OnObjectHighlightStart(hb.target.gameObject)) return;
 | |
|                     }
 | |
|                 }
 | |
|             } else {
 | |
|                 if (hb.highlighted) {
 | |
|                     if (OnObjectHighlightEnd != null && hb.target != null) {
 | |
|                         OnObjectHighlightEnd(hb.target.gameObject);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             if (selectOnClick || hb.isSelected) {
 | |
|                 if (hb.isSelected) {
 | |
|                     if (state && selectedAndHighlightedProfile != null) {
 | |
|                         selectedAndHighlightedProfile.Load(hb);
 | |
|                     } else if (selectedProfile != null) {
 | |
|                         selectedProfile.Load(hb);
 | |
|                     } else {
 | |
|                         hb.previousSettings.Load(hb);
 | |
|                     }
 | |
|                     if (hb.highlighted) {
 | |
|                         hb.UpdateMaterialProperties();
 | |
|                     } else {
 | |
|                         hb.SetHighlighted(true);
 | |
|                     }
 | |
|                     return;
 | |
|                 } else if (!highlightOnHover) {
 | |
|                     hb.SetHighlighted(false);
 | |
|                     return;
 | |
|                 }
 | |
|             }
 | |
|             hb.SetHighlighted(state);
 | |
|         }
 | |
| 
 | |
| 
 | |
|         void ToggleSelection() {
 | |
| 
 | |
|             HighlightManager.lastTriggerFrame = Time.frameCount;
 | |
| 
 | |
|             bool newState = toggle ? !hb.isSelected : true;
 | |
|             if (newState) {
 | |
|                 if (OnObjectSelected != null && !OnObjectSelected(gameObject)) return;
 | |
|             } else {
 | |
|                 if (OnObjectUnSelected != null && !OnObjectUnSelected(gameObject)) return;
 | |
|             }
 | |
| 
 | |
|             if (singleSelection && newState) {
 | |
|                 HighlightManager.DeselectAll();
 | |
|             }
 | |
|             hb.isSelected = newState;
 | |
|             if (newState && !HighlightManager.selectedObjects.Contains(hb)) {
 | |
|                 HighlightManager.selectedObjects.Add(hb);
 | |
|             } else if (!newState && HighlightManager.selectedObjects.Contains(hb)) {
 | |
|                 HighlightManager.selectedObjects.Remove(hb);
 | |
|             }
 | |
| 
 | |
|             if (hb.isSelected) {
 | |
|                 if (hb.previousSettings == null) {
 | |
|                     hb.previousSettings = ScriptableObject.CreateInstance<HighlightProfile>();
 | |
|                 }
 | |
|                 hb.previousSettings.Save(hb);
 | |
|             } else {
 | |
|                 hb.RestorePreviousHighlightEffectSettings();
 | |
|             }
 | |
| 
 | |
|             Highlight(true);
 | |
|         }
 | |
| 
 | |
|         public void OnTriggerEnter(Collider other) {
 | |
|             if (triggerMode == TriggerMode.Volume) {
 | |
|                 if ((volumeLayerMask & (1 << other.gameObject.layer)) != 0) {
 | |
|                     Highlight(true);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void OnTriggerExit(Collider other) {
 | |
|             if (triggerMode == TriggerMode.Volume) {
 | |
|                 if ((volumeLayerMask & (1 << other.gameObject.layer)) != 0) {
 | |
|                     Highlight(false);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|     }
 | |
| 
 | |
| } |