#if !UNITY_2019_3_OR_NEWER
#define CINEMACHINE_PHYSICS
#define CINEMACHINE_PHYSICS_2D
#endif
using UnityEngine;
namespace Cinemachine
{
/// An ad-hoc collection of helpers, used by Cinemachine
/// or its editor tools in various places
[DocumentationSorting(DocumentationSortingAttribute.Level.Undoc)]
public static class RuntimeUtility
{
/// Convenience to destroy an object, using the appropriate method depending
/// on whether the game is playing
/// The object to destroy
public static void DestroyObject(UnityEngine.Object obj)
{
if (obj != null)
{
#if UNITY_EDITOR
if (Application.isPlaying)
UnityEngine.Object.Destroy(obj);
else
UnityEngine.Object.DestroyImmediate(obj);
#else
UnityEngine.Object.Destroy(obj);
#endif
}
}
///
/// Check whether a GameObject is a prefab.
/// For editor only - some things are disallowed if prefab. In runtime, will always return false.
///
/// the object to check
/// If editor, checks if object is a prefab or prefab instance.
/// In runtime, returns false always
public static bool IsPrefab(GameObject gameObject)
{
#if UNITY_EDITOR
return UnityEditor.PrefabUtility.GetPrefabInstanceStatus(gameObject)
!= UnityEditor.PrefabInstanceStatus.NotAPrefab;
#else
return false;
#endif
}
#if CINEMACHINE_PHYSICS
private static RaycastHit[] s_HitBuffer = new RaycastHit[16];
private static int[] s_PenetrationIndexBuffer = new int[16];
///
/// Perform a raycast, but pass through any objects that have a given tag
///
/// The ray to cast
/// The returned results
/// Length of the raycast
/// Layers to include
/// Tag to ignore
/// True if something was hit. Results in hitInfo
public static bool RaycastIgnoreTag(
Ray ray, out RaycastHit hitInfo, float rayLength, int layerMask, in string ignoreTag)
{
if (ignoreTag.Length == 0)
{
if (Physics.Raycast(
ray, out hitInfo, rayLength, layerMask,
QueryTriggerInteraction.Ignore))
{
return true;
}
}
else
{
int closestHit = -1;
int numHits = Physics.RaycastNonAlloc(
ray, s_HitBuffer, rayLength, layerMask, QueryTriggerInteraction.Ignore);
for (int i = 0; i < numHits; ++i)
{
if (s_HitBuffer[i].collider.CompareTag(ignoreTag))
continue;
if (closestHit < 0 || s_HitBuffer[i].distance < s_HitBuffer[closestHit].distance)
closestHit = i;
}
if (closestHit >= 0)
{
hitInfo = s_HitBuffer[closestHit];
if (numHits == s_HitBuffer.Length)
s_HitBuffer = new RaycastHit[s_HitBuffer.Length * 2]; // full! grow for next time
return true;
}
}
hitInfo = new RaycastHit();
return false;
}
///
/// Perform a sphere cast, but pass through objects with a given tag
///
/// Start of the ray
/// Radius of the sphere cast
/// Normalized direction of the ray
/// Results go here
/// Length of the ray
/// Layers to include
/// Tag to ignore
/// True if something is hit. Results in hitInfo.
public static bool SphereCastIgnoreTag(
Vector3 rayStart, float radius, Vector3 dir,
out RaycastHit hitInfo, float rayLength,
int layerMask, in string ignoreTag)
{
int closestHit = -1;
int numPenetrations = 0;
float penetrationDistanceSum = 0;
int numHits = Physics.SphereCastNonAlloc(
rayStart, radius, dir, s_HitBuffer, rayLength, layerMask,
QueryTriggerInteraction.Ignore);
for (int i = 0; i < numHits; ++i)
{
var h = s_HitBuffer[i];
if (ignoreTag.Length > 0 && h.collider.CompareTag(ignoreTag))
continue;
// Collect overlapping items
if (h.distance == 0 && h.normal == -dir)
{
if (s_PenetrationIndexBuffer.Length > numPenetrations + 1)
s_PenetrationIndexBuffer[numPenetrations++] = i;
// hitInfo for overlapping colliders will have special
// values that are not helpful to the caller. Fix that here.
var scratchCollider = GetScratchCollider();
scratchCollider.radius = radius;
var c = h.collider;
if (Physics.ComputePenetration(
scratchCollider, rayStart, Quaternion.identity,
c, c.transform.position, c.transform.rotation,
out var offsetDir, out var offsetDistance))
{
h.point = rayStart + offsetDir * (offsetDistance - radius);
h.distance = offsetDistance - radius; // will be -ve
h.normal = offsetDir;
s_HitBuffer[i] = h;
penetrationDistanceSum += h.distance;
}
else
{
continue; // don't know what's going on, just forget about it
}
}
if (closestHit < 0 || h.distance < s_HitBuffer[closestHit].distance)
{
closestHit = i;
}
}
// Naively combine penetrating items
if (numPenetrations > 1)
{
hitInfo = new RaycastHit();
for (int i = 0; i < numPenetrations; ++i)
{
var h = s_HitBuffer[s_PenetrationIndexBuffer[i]];
var t = h.distance / penetrationDistanceSum;
hitInfo.point += h.point * t;
hitInfo.distance += h.distance * t;
hitInfo.normal += h.normal * t;
}
hitInfo.normal = hitInfo.normal.normalized;
return true;
}
if (closestHit >= 0)
{
hitInfo = s_HitBuffer[closestHit];
if (numHits == s_HitBuffer.Length)
s_HitBuffer = new RaycastHit[s_HitBuffer.Length * 2]; // full! grow for next time
return true;
}
hitInfo = new RaycastHit();
return false;
}
private static SphereCollider s_ScratchCollider;
private static GameObject s_ScratchColliderGameObject;
internal static SphereCollider GetScratchCollider()
{
if (s_ScratchColliderGameObject == null)
{
s_ScratchColliderGameObject = new GameObject("Cinemachine Scratch Collider");
s_ScratchColliderGameObject.hideFlags = HideFlags.HideAndDontSave;
s_ScratchColliderGameObject.transform.position = Vector3.zero;
s_ScratchColliderGameObject.SetActive(true);
s_ScratchCollider = s_ScratchColliderGameObject.AddComponent();
s_ScratchCollider.isTrigger = true;
var rb = s_ScratchColliderGameObject.AddComponent();
rb.detectCollisions = false;
rb.isKinematic = true;
}
return s_ScratchCollider;
}
internal static void DestroyScratchCollider()
{
if (s_ScratchColliderGameObject != null)
{
s_ScratchColliderGameObject.SetActive(false);
DestroyObject(s_ScratchColliderGameObject.GetComponent());
}
DestroyObject(s_ScratchCollider);
DestroyObject(s_ScratchColliderGameObject);
s_ScratchColliderGameObject = null;
s_ScratchCollider = null;
}
#endif
///
/// Normalize a curve so that its X and Y axes range from 0 to 1
///
/// Curve to normalize
/// If true, normalize the X axis
/// If true, normalize the Y axis
/// The normalized curve
public static AnimationCurve NormalizeCurve(
AnimationCurve curve, bool normalizeX, bool normalizeY)
{
if (!normalizeX && !normalizeY)
return curve;
Keyframe[] keys = curve.keys;
if (keys.Length > 0)
{
float minTime = keys[0].time;
float maxTime = minTime;
float minVal = keys[0].value;
float maxVal = minVal;
for (int i = 0; i < keys.Length; ++i)
{
minTime = Mathf.Min(minTime, keys[i].time);
maxTime = Mathf.Max(maxTime, keys[i].time);
minVal = Mathf.Min(minVal, keys[i].value);
maxVal = Mathf.Max(maxVal, keys[i].value);
}
float range = maxTime - minTime;
float timeScale = range < 0.0001f ? 1 : 1 / range;
range = maxVal - minVal;
float valScale = range < 1 ? 1 : 1 / range;
float valOffset = 0;
if (range < 1)
{
if (minVal > 0 && minVal + range <= 1)
valOffset = minVal;
else
valOffset = 1 - range;
}
for (int i = 0; i < keys.Length; ++i)
{
if (normalizeX)
keys[i].time = (keys[i].time - minTime) * timeScale;
if (normalizeY)
keys[i].value = ((keys[i].value - minVal) * valScale) + valOffset;
}
curve.keys = keys;
}
return curve;
}
}
}