using System; using System.Collections.Generic; using Cinemachine.Utility; using UnityEngine; namespace Cinemachine { using IntPoint = ClipperLib.IntPoint; using Clipper = ClipperLib.Clipper; using ClipperOffset = ClipperLib.ClipperOffset; using ClipperBase = ClipperLib.ClipperBase; using IntRect = ClipperLib.IntRect; using JoinType = ClipperLib.JoinType; using EndType = ClipperLib.EndType; using PolyType = ClipperLib.PolyType; using PolyFillType = ClipperLib.PolyFillType; using ClipType = ClipperLib.ClipType; /// /// Responsible for baking confiners via BakeConfiner function. /// internal class ConfinerOven { public class BakedSolution { public float FrustumHeight { get; } private float m_frustumSizeIntSpace; private readonly AspectStretcher m_AspectStretcher; private readonly bool m_HasBones; private readonly double m_SqrPolygonDiagonal; private List> m_OriginalPolygon; public List> m_Solution; private const double k_ClipperEpsilon = 0.01f * k_FloatToIntScaler; public BakedSolution( float aspectRatio, float frustumHeight, bool hasBones, Rect polygonBounds, List> originalPolygon, List> solution) { m_AspectStretcher = new AspectStretcher(aspectRatio, polygonBounds.center.x); FrustumHeight = frustumHeight; m_frustumSizeIntSpace = frustumHeight * k_FloatToIntScaler; m_HasBones = hasBones; m_OriginalPolygon = originalPolygon; m_Solution = solution; float polygonSizeX = polygonBounds.width / aspectRatio * k_FloatToIntScaler; float polygonSizeY = polygonBounds.height * k_FloatToIntScaler; m_SqrPolygonDiagonal = polygonSizeX * polygonSizeX + polygonSizeY * polygonSizeY; } public void Clear() { m_Solution = null; m_OriginalPolygon = null; } public bool IsValid(float frustumHeight) { return m_Solution != null && Mathf.Abs(frustumHeight - FrustumHeight) < k_MinStepSize; } public Vector2 ConfinePoint(in Vector2 pointToConfine) { if (m_Solution.Count <= 0) return pointToConfine; // empty confiner -> no need to confine Vector2 pInConfinerSpace = m_AspectStretcher.Stretch(pointToConfine); IntPoint p = new IntPoint(pInConfinerSpace.x * k_FloatToIntScaler, pInConfinerSpace.y * k_FloatToIntScaler); for (int i = 0; i < m_Solution.Count; ++i) { if (Clipper.PointInPolygon(p, m_Solution[i]) != 0) // 0: outside, +1: inside , -1: point on poly boundary { return pointToConfine; // inside, no confinement needed } } // If the poly has bones and if the position to confine is not outside of the original // bounding shape, then it is possible that the bone in a neighbouring section // is closer than the bone in the correct section of the polygon, if the current section // is very large and the neighbouring section is small. In that case, we'll need to // add an extra check when calculating the nearest point. bool checkIntersectOriginal = m_HasBones && IsInsideOriginal(p); // Confine point IntPoint closest = p; double minDistance = double.MaxValue; for (int i = 0; i < m_Solution.Count; ++i) { int numPoints = m_Solution[i].Count; for (int j = 0; j < numPoints; ++j) { IntPoint l1 = m_Solution[i][j]; IntPoint l2 = m_Solution[i][(j + 1) % numPoints]; IntPoint c = IntPointLerp(l1, l2, ClosestPointOnSegment(p, l1, l2)); double diffX = Mathf.Abs(p.X - c.X); double diffY = Mathf.Abs(p.Y - c.Y); double distance = diffX * diffX + diffY * diffY; // penalty for points from which the target is not visible, preferring visibility over proximity if (diffX > m_frustumSizeIntSpace || diffY > m_frustumSizeIntSpace) { distance += m_SqrPolygonDiagonal; // penalty is the biggest distance between any two points } if (distance < minDistance && (!checkIntersectOriginal || !DoesIntersectOriginal(p, c))) { minDistance = distance; closest = c; } } } var result = new Vector2(closest.X * k_IntToFloatScaler, closest.Y * k_IntToFloatScaler); return m_AspectStretcher.Unstretch(result); } #if UNITY_EDITOR // Used by inspector to draw the baked path private List> m_Vector2Path; public List> GetBakedPath() { // Convert to client space var numPaths = m_Solution.Count; if (m_Vector2Path == null) { m_Vector2Path = new List>(numPaths); for (int i = 0; i < numPaths; ++i) { var srcPoly = m_Solution[i]; int numPoints = srcPoly.Count; var pathSegment = new List(numPoints); for (int j = 0; j < numPoints; j++) { // Restore the original aspect ratio pathSegment.Add(m_AspectStretcher.Unstretch( new Vector2(srcPoly[j].X, srcPoly[j].Y) * k_IntToFloatScaler)); } m_Vector2Path.Add(pathSegment); } } return m_Vector2Path; } #endif private bool IsInsideOriginal(IntPoint p) { for (int i = 0; i < m_OriginalPolygon.Count; ++i) { if (Clipper.PointInPolygon(p, m_OriginalPolygon[i]) != 0) { return true; } } return false; } private static float ClosestPointOnSegment(IntPoint p, IntPoint s0, IntPoint s1) { double sX = s1.X - s0.X; double sY = s1.Y - s0.Y; double len2 = sX * sX + sY * sY; if (len2 < k_ClipperEpsilon) return 0; // degenerate segment double s0pX = p.X - s0.X; double s0pY = p.Y - s0.Y; double dot = s0pX * sX + s0pY * sY; return Mathf.Clamp01((float) (dot / len2)); } private static IntPoint IntPointLerp(IntPoint a, IntPoint b, float lerp) { return new IntPoint { X = Mathf.RoundToInt(a.X + (b.X - a.X) * lerp), Y = Mathf.RoundToInt(a.Y + (b.Y - a.Y) * lerp), }; } private bool DoesIntersectOriginal(IntPoint l1, IntPoint l2) { foreach (var original in m_OriginalPolygon) { int numPoints = original.Count; for (int i = 0; i < numPoints; ++i) { if (FindIntersection(l1, l2, original[i], original[(i + 1) % numPoints]) == 2) { return true; } } } return false; } private static int FindIntersection(in IntPoint p1, in IntPoint p2, in IntPoint p3, in IntPoint p4) { // Get the segments' parameters. double dx12 = p2.X - p1.X; double dy12 = p2.Y - p1.Y; double dx34 = p4.X - p3.X; double dy34 = p4.Y - p3.Y; // Solve for t1 and t2 double denominator = dy12 * dx34 - dx12 * dy34; double t1 = ((p1.X - p3.X) * dy34 + (p3.Y - p1.Y) * dx34) / denominator; if (double.IsInfinity(t1) || double.IsNaN(t1)) { // The lines are parallel (or close enough to it). if (IntPointDiffSqrMagnitude(p1, p3) < k_ClipperEpsilon || IntPointDiffSqrMagnitude(p1, p4) < k_ClipperEpsilon || IntPointDiffSqrMagnitude(p2, p3) < k_ClipperEpsilon || IntPointDiffSqrMagnitude(p2, p4) < k_ClipperEpsilon) { return 2; // they are the same line, or very close parallels } return 0; // no intersection } double t2 = ((p3.X - p1.X) * dy12 + (p1.Y - p3.Y) * dx12) / -denominator; return (t1 >= 0 && t1 <= 1 && t2 >= 0 && t2 < 1) ? 2 : 1; // 2 = segments intersect, 1 = lines intersect } private static double IntPointDiffSqrMagnitude(IntPoint p1, IntPoint p2) { double X = p1.X - p2.X; double Y = p1.Y - p2.Y; return X * X + Y * Y; } } private struct AspectStretcher { public float Aspect { get; } private readonly float m_InverseAspect; private readonly float m_CenterX; public AspectStretcher(float aspect, float centerX) { Aspect = aspect; m_InverseAspect = 1 / Aspect; m_CenterX = centerX; } public Vector2 Stretch(Vector2 p) { return new Vector2((p.x - m_CenterX) * m_InverseAspect + m_CenterX, p.y); } public Vector2 Unstretch(Vector2 p) { return new Vector2((p.x - m_CenterX) * Aspect + m_CenterX, p.y); } } private float m_MinFrustumHeightWithBones; private List> m_OriginalPolygon; private List> m_Skeleton = new List>(); private const long k_FloatToIntScaler = 100000; private const float k_IntToFloatScaler = 1.0f / k_FloatToIntScaler; private const float k_MinStepSize = 0.005f; private Rect m_PolygonRect; AspectStretcher m_AspectStretcher = new AspectStretcher(1, 0); private float m_maxComputationTimeForFullSkeletonBakeInSeconds = 5f; public ConfinerOven( in List> inputPath, in float aspectRatio, float maxFrustumHeight) : base() { Initialize(inputPath, aspectRatio, maxFrustumHeight); } /// /// Converts and returns a prebaked ConfinerState for the input frustumHeight. /// public BakedSolution GetBakedSolution(float frustumHeight) { // Inflate with clipper to frustumHeight var offsetter = new ClipperOffset(); offsetter.AddPaths(m_OriginalPolygon, JoinType.jtMiter, EndType.etClosedPolygon); var solution = new List>(); offsetter.Execute(ref solution, -1f * frustumHeight * k_FloatToIntScaler); // Add in the skeleton var bakedSolution = new List>(); if (State == BakingState.BAKING || m_Skeleton.Count == 0) { bakedSolution = solution; } else { Clipper c = new Clipper(); c.AddPaths(solution, PolyType.ptSubject, true); c.AddPaths(m_Skeleton, PolyType.ptClip, true); c.Execute(ClipType.ctUnion, bakedSolution, PolyFillType.pftEvenOdd, PolyFillType.pftEvenOdd); } return new BakedSolution( m_AspectStretcher.Aspect, frustumHeight, m_MinFrustumHeightWithBones < frustumHeight, m_PolygonRect, m_OriginalPolygon, bakedSolution); } private struct PolygonSolution { public List> m_Polygons; public float m_FrustumHeight; public bool StateChanged(in List> paths) { if (paths.Count != m_Polygons.Count) return true; for (int i = 0; i < paths.Count; ++i) { if (paths[i].Count != m_Polygons[i].Count) return true; } return false; } public bool IsEmpty => m_Polygons == null; } public enum BakingState { BAKING, BAKED, TIMEOUT } public BakingState State { get; private set; } public float m_BakeProgress; private struct BakingStateCache { public ClipperOffset offsetter; public List solutions; public PolygonSolution rightCandidate; public PolygonSolution leftCandidate; public List> maxCandidate; public float stepSize; public float maxFrustumHeight; public float currentFrustumHeight; public float bakeTime; } private BakingStateCache m_Cache; private void Initialize(in List> inputPath, in float aspectRatio, float maxFrustumHeight) { m_Skeleton.Clear(); m_Cache.maxFrustumHeight = maxFrustumHeight; m_MinFrustumHeightWithBones = float.MaxValue; m_PolygonRect = GetPolygonBoundingBox(inputPath); m_AspectStretcher = new AspectStretcher(aspectRatio, m_PolygonRect.center.x); // Initialize clipper m_OriginalPolygon = new List>(inputPath.Count); for (var i = 0; i < inputPath.Count; ++i) { var srcPath = inputPath[i]; int numPoints = srcPath.Count; var path = new List(numPoints); for (int j = 0; j < numPoints; ++j) { // Neutralize the aspect ratio var p = m_AspectStretcher.Stretch(srcPath[j]); path.Add(new IntPoint(p.x * k_FloatToIntScaler, p.y * k_FloatToIntScaler)); } m_OriginalPolygon.Add(path); } // Skip the expensive skeleton calculation if it's not wanted if (m_Cache.maxFrustumHeight < 0) { State = BakingState.BAKED; // if we don't need skeleton, then we don't need to bake return; } // Don't compute further than what is the theoretical max float polygonHalfHeight = Mathf.Min(m_PolygonRect.width / aspectRatio, m_PolygonRect.height) / 2f; // exact comparison to 0 is intentional! if (m_Cache.maxFrustumHeight == 0 || m_Cache.maxFrustumHeight > polygonHalfHeight) { m_Cache.maxFrustumHeight = polygonHalfHeight; } m_Cache.stepSize = m_Cache.maxFrustumHeight; // Binary search for state changes so we can compute the skeleton m_Cache.offsetter = new ClipperOffset(); m_Cache.offsetter.AddPaths(m_OriginalPolygon, JoinType.jtMiter, EndType.etClosedPolygon); List> solution = new List>(); m_Cache.offsetter.Execute(ref solution, 0); m_Cache.solutions = new List(); m_Cache.solutions.Add(new PolygonSolution { m_Polygons = solution, m_FrustumHeight = 0, }); m_Cache.rightCandidate = new PolygonSolution(); m_Cache.leftCandidate = new PolygonSolution { m_Polygons = solution, m_FrustumHeight = 0, }; m_Cache.currentFrustumHeight = 0; m_Cache.maxCandidate = new List>(); m_Cache.offsetter.Execute(ref m_Cache.maxCandidate, -1f * m_Cache.maxFrustumHeight * k_FloatToIntScaler); m_Cache.bakeTime = 0; State = BakingState.BAKING; m_BakeProgress = 0; } /// /// Creates shrinkable polygons from input parameters. /// The algorithm is divide and conquer. It iteratively shrinks down the input /// polygon towards its shrink directions. If the polygon intersects with itself, /// then we divide the polygon into two polygons at the intersection point, and /// continue the algorithm on these two polygons separately. We need to keep track of /// the connectivity information between sub-polygons. /// public void BakeConfiner(float maxComputationTimePerFrameInSeconds) { if (State != BakingState.BAKING) return; var startTime = Time.realtimeSinceStartup; while (m_Cache.solutions.Count < 1000) { bool stateChangeFound = false; var numPaths = m_Cache.leftCandidate.m_Polygons.Count; var candidate = new List>(numPaths); m_Cache.stepSize = Mathf.Min(m_Cache.stepSize, m_Cache.maxFrustumHeight - m_Cache.leftCandidate.m_FrustumHeight); #if false Debug.Log($"States = {solutions.Count}, " + $"Frustum height = {currentFrustumHeight}, stepSize = {stepSize}"); #endif m_Cache.currentFrustumHeight = m_Cache.leftCandidate.m_FrustumHeight + m_Cache.stepSize; if (Math.Abs(m_Cache.currentFrustumHeight - m_Cache.maxFrustumHeight) < UnityVectorExtensions.Epsilon) { candidate = m_Cache.maxCandidate; } else { m_Cache.offsetter.Execute( ref candidate, -1f * m_Cache.currentFrustumHeight * k_FloatToIntScaler); } stateChangeFound = m_Cache.leftCandidate.StateChanged(in candidate); if (stateChangeFound) { m_Cache.rightCandidate = new PolygonSolution { m_Polygons = candidate, m_FrustumHeight = m_Cache.currentFrustumHeight, }; m_Cache.stepSize = Mathf.Max(m_Cache.stepSize / 2f, k_MinStepSize); } else { m_Cache.leftCandidate = new PolygonSolution { m_Polygons = candidate, m_FrustumHeight = m_Cache.currentFrustumHeight, }; // if we have not found right yet, then we don't need to decrease stepsize if (!m_Cache.rightCandidate.IsEmpty) { m_Cache.stepSize = Mathf.Max(m_Cache.stepSize / 2f, k_MinStepSize); } } // if we have a right candidate, and left and right are sufficiently close, // then we have located a state change point if (!m_Cache.rightCandidate.IsEmpty && m_Cache.stepSize <= k_MinStepSize) { // Add both states: one before the state change and one after m_Cache.solutions.Add(m_Cache.leftCandidate); m_Cache.solutions.Add(m_Cache.rightCandidate); m_Cache.leftCandidate = m_Cache.rightCandidate; m_Cache.rightCandidate = new PolygonSolution(); // Back to max step m_Cache.stepSize = m_Cache.maxFrustumHeight; } else if (m_Cache.rightCandidate.IsEmpty || m_Cache.leftCandidate.m_FrustumHeight >= m_Cache.maxFrustumHeight) { m_Cache.solutions.Add(m_Cache.leftCandidate); break; // stop searching, because we are at the bound } // Pause after max time per iteration reached var elapsedTime = Time.realtimeSinceStartup - startTime; if (elapsedTime > maxComputationTimePerFrameInSeconds) { m_Cache.bakeTime += elapsedTime; if (m_Cache.bakeTime > m_maxComputationTimeForFullSkeletonBakeInSeconds) { State = BakingState.TIMEOUT; } m_BakeProgress = m_Cache.leftCandidate.m_FrustumHeight / m_Cache.maxFrustumHeight; return; } } ComputeSkeleton(in m_Cache.solutions); m_BakeProgress = 1; State = BakingState.BAKED; } private static Rect GetPolygonBoundingBox(in List> polygons) { float minX = float.PositiveInfinity, maxX = float.NegativeInfinity; float minY = float.PositiveInfinity, maxY = float.NegativeInfinity; for (int i = 0; i < polygons.Count; ++i) { var path = polygons[i]; for (int j = 0; j < path.Count; ++j) { var p = path[j]; minX = Mathf.Min(minX, p.x); maxX = Mathf.Max(maxX, p.x); minY = Mathf.Min(minY, p.y); maxY = Mathf.Max(maxY, p.y); } } return new Rect(minX, minY, Mathf.Max(0, maxX - minX), Mathf.Max(0, maxY - minY)); } private void ComputeSkeleton(in List solutions) { // At each state change point, collect geometry that gets lost over the transition var clipper = new Clipper(); var offsetter = new ClipperOffset(); for (int i = 1; i < solutions.Count - 1; i += 2) { var prev = solutions[i]; var next = solutions[i+1]; const int padding = 5; // to counteract precision problems - inflates small regions double step = padding * k_FloatToIntScaler * (next.m_FrustumHeight - prev.m_FrustumHeight); // Grow the larger polygon to inflate marginal regions var expandedPrev = new List>(); offsetter.Clear(); offsetter.AddPaths(prev.m_Polygons, JoinType.jtMiter, EndType.etClosedPolygon); offsetter.Execute(ref expandedPrev, step); // Grow the smaller polygon to be a bit bigger than the expanded larger one var expandedNext = new List>(); offsetter.Clear(); offsetter.AddPaths(next.m_Polygons, JoinType.jtMiter, EndType.etClosedPolygon); offsetter.Execute(ref expandedNext, step * 2); // Compute the difference - this is the lost geometry var solution = new List>(); clipper.Clear(); clipper.AddPaths(expandedPrev, PolyType.ptSubject, true); clipper.AddPaths(expandedNext, PolyType.ptClip, true); clipper.Execute(ClipType.ctDifference, solution, PolyFillType.pftEvenOdd, PolyFillType.pftEvenOdd); // Add that lost geometry to the skeleton if (solution.Count > 0 && solution[0].Count > 0) { m_Skeleton.AddRange(solution); if (m_MinFrustumHeightWithBones == float.MaxValue) m_MinFrustumHeightWithBones = next.m_FrustumHeight; } } } } }