// Copyright 2016 Google Inc. All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. using System.Collections.Generic; using UnityEngine; using UnityEngine.EventSystems; using UnityEngine.UI; /// This script provides a raycaster for use with the GvrPointerInputModule. /// It behaves similarly to the standards Graphic raycaster, except that it utilize raycast /// modes specifically for Gvr. /// /// View GvrBasePointerRaycaster.cs and GvrPointerInputModule.cs for more details. [AddComponentMenu("GoogleVR/GvrPointerGraphicRaycaster")] [RequireComponent(typeof(Canvas))] public class GvrPointerGraphicRaycaster : GvrBasePointerRaycaster { public enum BlockingObjects { None = 0, TwoD = 1, ThreeD = 2, All = 3, } private const int NO_EVENT_MASK_SET = -1; public bool ignoreReversedGraphics = true; public BlockingObjects blockingObjects = BlockingObjects.None; public LayerMask blockingMask = NO_EVENT_MASK_SET; private Canvas targetCanvas; private List raycastResults = new List(); private Camera cachedPointerEventCamera; private static readonly List sortedGraphics = new List(); public override Camera eventCamera { get { switch (raycastMode) { case RaycastMode.Direct: if (cachedPointerEventCamera == null) { if (!IsPointerAvailable()) { Debug.LogError("Can't find eventCamera, pointer is not available."); return Camera.main; } Transform pointerTransform = GvrPointerManager.Pointer.PointerTransform; cachedPointerEventCamera = pointerTransform.GetComponent(); if (cachedPointerEventCamera == null) { Debug.LogError("GvrPointerGraphicRaycaster requires GvrPointer to have a Camera when in Direct mode."); } } return cachedPointerEventCamera != null ? cachedPointerEventCamera : Camera.main; case RaycastMode.Camera: default: return Camera.main; } } } private Canvas canvas { get { if (targetCanvas != null) return targetCanvas; targetCanvas = GetComponent(); return targetCanvas; } } protected GvrPointerGraphicRaycaster() { } public override void Raycast(PointerEventData eventData, List resultAppendList) { if (canvas == null) { return; } if (!IsPointerAvailable() || eventCamera == null) { return; } if (canvas.renderMode != RenderMode.WorldSpace) { Debug.LogError("GvrPointerGraphicRaycaster requires that the canvas renderMode is set to WorldSpace."); return; } Ray ray = GetRay(); float hitDistance = float.MaxValue; if (blockingObjects != BlockingObjects.None) { float dist = eventCamera.farClipPlane - eventCamera.nearClipPlane; if (blockingObjects == BlockingObjects.ThreeD || blockingObjects == BlockingObjects.All) { RaycastHit hit; if (Physics.Raycast(ray, out hit, dist, blockingMask)) { hitDistance = hit.distance; } } if (blockingObjects == BlockingObjects.TwoD || blockingObjects == BlockingObjects.All) { RaycastHit2D hit = Physics2D.Raycast(ray.origin, ray.direction, dist, blockingMask); if (hit.collider != null) { hitDistance = hit.fraction * dist; } } } raycastResults.Clear(); Ray finalRay; Raycast(canvas, ray, eventCamera, MaxPointerDistance, raycastResults, out finalRay); for (int index = 0; index < raycastResults.Count; index++) { GameObject go = raycastResults[index].gameObject; bool appendGraphic = true; if (ignoreReversedGraphics) { // If we have a camera compare the direction against the cameras forward. Vector3 cameraFoward = eventCamera.transform.rotation * Vector3.forward; Vector3 dir = go.transform.rotation * Vector3.forward; appendGraphic = Vector3.Dot(cameraFoward, dir) > 0; } if (appendGraphic) { float distance = 0; Transform trans = go.transform; Vector3 transForward = trans.forward; // http://geomalgorithms.com/a06-_intersect-2.html float transDot = Vector3.Dot(transForward, trans.position - finalRay.origin); float rayDot = Vector3.Dot(transForward, finalRay.direction); distance = transDot / rayDot; // Check to see if the go is behind the camera. if (distance < 0) { continue; } if (distance >= hitDistance) { continue; } Vector3 hitPosition = finalRay.origin + (finalRay.direction * distance); RaycastResult castResult = new RaycastResult { gameObject = go, module = this, distance = distance, worldPosition = hitPosition, screenPosition = eventCamera.WorldToScreenPoint(hitPosition), index = resultAppendList.Count, depth = raycastResults[index].depth, sortingLayer = canvas.sortingLayerID, sortingOrder = canvas.sortingOrder }; resultAppendList.Add(castResult); } } } /// Perform a raycast into the screen and collect all graphics underneath it. private static void Raycast(Canvas canvas, Ray ray, Camera cam, float maxPointerDistance, List results, out Ray finalRay) { Vector3 screenPoint = cam.WorldToScreenPoint(ray.GetPoint(maxPointerDistance)); finalRay = cam.ScreenPointToRay(screenPoint); // Necessary for the event system IList foundGraphics = GraphicRegistry.GetGraphicsForCanvas(canvas); for (int i = 0; i < foundGraphics.Count; ++i) { Graphic graphic = foundGraphics[i]; // -1 means it hasn't been processed by the canvas, which means it isn't actually drawn if (graphic.depth == -1 || !graphic.raycastTarget) { continue; } if (!RectTransformUtility.RectangleContainsScreenPoint(graphic.rectTransform, screenPoint, cam)) { continue; } if (graphic.Raycast(screenPoint, cam)) { sortedGraphics.Add(graphic); } } sortedGraphics.Sort((g1, g2) => g2.depth.CompareTo(g1.depth)); for (int i = 0; i < sortedGraphics.Count; ++i) { results.Add(sortedGraphics[i]); } sortedGraphics.Clear(); } }