using System.Collections; using System.Collections.Generic; using UnityEngine; public abstract class FollowerBase : MonoBehaviour { [SerializeField, Range(0.1f, 10.0f)] private float windowDistance = 0.5f; public float WindowDistance { get { return windowDistance; } set { windowDistance = Mathf.Abs(value); } } [SerializeField, Range(0.0f, 100.0f), Tooltip("How quickly to interpolate the window towards its target position and rotation.")] private float windowFollowSpeed = 1.2f; public float WindowFollowSpeed { get { return windowFollowSpeed; } set { windowFollowSpeed = Mathf.Abs(value); } } [Header("Window Settings")] [SerializeField, Tooltip("What part of the view port to anchor the window to.")] private TextAnchor windowAnchor = TextAnchor.LowerCenter; public TextAnchor WindowAnchor { get { return windowAnchor; } set { windowAnchor = value; } } [SerializeField, Tooltip("The offset from the view port center applied based on the window anchor selection.")] private Vector2 windowOffset = new Vector2(0.1f, 0.1f); public Vector2 WindowOffset { get { return windowOffset; } set { windowOffset = value; } } [SerializeField] private Vector2 defaultWindowRotation = new Vector2(10.0f, 20.0f); public Vector2 DefaultWindowRotation { get { return defaultWindowRotation; } set { defaultWindowRotation = value; } } private Quaternion windowHorizontalRotation; private Quaternion windowHorizontalRotationInverse; private Quaternion windowVerticalRotation; private Quaternion windowVerticalRotationInverse; protected virtual void OnEnable() { windowHorizontalRotation = Quaternion.AngleAxis(defaultWindowRotation.y, Vector3.right); windowHorizontalRotationInverse = Quaternion.Inverse(windowHorizontalRotation); windowVerticalRotation = Quaternion.AngleAxis(defaultWindowRotation.x, Vector3.up); windowVerticalRotationInverse = Quaternion.Inverse(windowVerticalRotation); } protected virtual Vector3 CalculateWindowPosition(Transform cameraTransform) { Vector3 position = cameraTransform.position + (cameraTransform.forward * WindowDistance); Vector3 horizontalOffset = cameraTransform.right * windowOffset.x; Vector3 verticalOffset = cameraTransform.up * windowOffset.y; switch (windowAnchor) { case TextAnchor.UpperLeft: position += verticalOffset - horizontalOffset; break; case TextAnchor.UpperCenter: position += verticalOffset; break; case TextAnchor.UpperRight: position += verticalOffset + horizontalOffset; break; case TextAnchor.MiddleLeft: position -= horizontalOffset; break; case TextAnchor.MiddleRight: position += horizontalOffset; break; case TextAnchor.LowerLeft: position -= verticalOffset + horizontalOffset; break; case TextAnchor.LowerCenter: position -= verticalOffset; break; case TextAnchor.LowerRight: position -= verticalOffset - horizontalOffset; break; } return position; } protected virtual Quaternion CalculateWindowRotation(Transform cameraTransform) { Quaternion rotation = cameraTransform.rotation; switch (windowAnchor) { case TextAnchor.UpperLeft: rotation *= windowHorizontalRotationInverse * windowVerticalRotationInverse; break; case TextAnchor.UpperCenter: rotation *= windowHorizontalRotationInverse; break; case TextAnchor.UpperRight: rotation *= windowHorizontalRotationInverse * windowVerticalRotation; break; case TextAnchor.MiddleLeft: rotation *= windowVerticalRotationInverse; break; case TextAnchor.MiddleRight: rotation *= windowVerticalRotation; break; case TextAnchor.LowerLeft: rotation *= windowHorizontalRotation * windowVerticalRotationInverse; break; case TextAnchor.LowerCenter: rotation *= windowHorizontalRotation; break; case TextAnchor.LowerRight: rotation *= windowHorizontalRotation * windowVerticalRotation; break; } return rotation; } protected virtual void LateUpdate() { Follow(); } protected abstract void Follow(); }