// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using Microsoft.MixedReality.Toolkit.Utilities; using System; using UnityEngine; namespace Microsoft.MixedReality.Toolkit.Boundary { /// /// The InscribedRectangle class defines the largest rectangle within an /// arbitrary shape. /// public class InscribedRectangle { /// /// Total number of starting points randomly generated within the boundary. /// private const int randomPointCount = 30; /// /// The total amount of height, in meters, we want to gain with each binary search /// change before we decide that it's good enough. /// private const float minimumHeightGain = 0.01f; /// /// Angles to use for fitting the rectangle within the boundary. /// private static readonly float[] FitAngles = { 0, 15, 30, 45, 60, 75, 90, 105, 120, 135, 150, 165 }; /// /// Aspect ratios used when fitting rectangles within the boundary. /// private static readonly float[] AspectRatios = { 1.0f, 1.5f, 2.0f, 2.5f, 3.0f, 3.5f, 4.0f, 4.5f, 5.0f, 5.5f, 6, 6.5f, 7, 7.5f, 8.0f, 8.5f, 9.0f, 9.5f, 10.0f, 10.5f, 11.0f, 11.5f, 12.0f, 12.5f, 13.0f, 13.5f, 14.0f, 14.5f, 15.0f}; /// /// The center point of the inscribed rectangle. /// public Vector2 Center { get; private set; } = EdgeUtilities.InvalidPoint; /// /// The width of the inscribed rectangle. /// public float Width { get; private set; } = 0f; /// /// The height of the inscribed rectangle. /// public float Height { get; private set; } = 0f; /// /// The rotation angle, in degrees, of the inscribed rectangle. /// public float Angle { get; private set; } = 0f; /// /// Is the described rectangle valid? /// /// /// A rectangle is considered valid if its center point is valid. /// public bool IsValid => EdgeUtilities.IsValidPoint(Center); /// /// Finds a large inscribed rectangle. Tries to be maximal but this is /// best effort. The algorithm used was inspired by the blog post /// https://d3plus.org/blog/behind-the-scenes/2014/07/08/largest-rect/ /// Random points within the polygon are chosen, and then 2 lines are /// drawn through those points. The midpoints of those lines are /// used as the center of various rectangles, using a binary search to /// vary the size, until the largest fit-able rectangle is found. /// This is then repeated for predefined angles (0-180 in steps of 15) /// and aspect ratios (1 to 15 in steps of 0.5). /// /// The boundary geometry. /// Random number generator seed. /// /// For the most reproducible results, use the same randomSeed value each time this method is called. /// public InscribedRectangle(Edge[] geometryEdges, int randomSeed) { if (geometryEdges == null || geometryEdges.Length == 0) { Debug.LogError("InscribedRectangle requires an array of Edges. You passed in a null or empty array."); return; } // Clear previous rectangle Center = EdgeUtilities.InvalidPoint; Width = 0; Height = 0; Angle = 0; float minX = EdgeUtilities.maxWidth; float minY = EdgeUtilities.maxWidth; float maxX = -EdgeUtilities.maxWidth; float maxY = -EdgeUtilities.maxWidth; // Find min x, min y, max x, max y for (int i = 0; i < geometryEdges.Length; i++) { Edge edge = geometryEdges[i]; if ((edge.PointA.x < minX) || (edge.PointB.x < minX)) { minX = Mathf.Min(edge.PointA.x, edge.PointB.x); } if ((edge.PointA.y < minY) || (edge.PointB.y < minY)) { minY = Mathf.Min(edge.PointA.y, edge.PointB.y); } if ((edge.PointA.x > maxX) || (edge.PointB.x > maxX)) { maxX = Mathf.Max(edge.PointA.x, edge.PointB.x); } if ((edge.PointA.y > maxY) || (edge.PointB.y > maxY)) { maxY = Mathf.Max(edge.PointA.y, edge.PointB.y); } } // Generate random points until we have randomPointCount starting points Vector2[] startingPoints = new Vector2[randomPointCount]; { System.Random random = new System.Random(randomSeed); for (int i = 0; i < startingPoints.Length; i++) { Vector2 candidatePoint; do { candidatePoint.x = ((float)random.NextDouble() * (maxX - minX)) + minX; candidatePoint.y = ((float)random.NextDouble() * (maxY - minY)) + minY; } while (!EdgeUtilities.IsInsideBoundary(geometryEdges, candidatePoint)); startingPoints[i] = candidatePoint; } } for (int angleIndex = 0; angleIndex < FitAngles.Length; angleIndex++) { for (int pointIndex = 0; pointIndex < startingPoints.Length; pointIndex++) { Vector2 topCollisionPoint; Vector2 bottomCollisionPoint; Vector2 leftCollisionPoint; Vector2 rightCollisionPoint; float angleRadians = MathUtilities.DegreesToRadians(FitAngles[angleIndex]); // Find the collision point of a cross through the given point at the given angle. // Note, we are ignoring the return value as we are checking each point's validity // individually. FindSurroundingCollisionPoints( geometryEdges, startingPoints[pointIndex], angleRadians, out topCollisionPoint, out bottomCollisionPoint, out leftCollisionPoint, out rightCollisionPoint); float newWidth; float newHeight; if (EdgeUtilities.IsValidPoint(topCollisionPoint) && EdgeUtilities.IsValidPoint(bottomCollisionPoint)) { float aX = topCollisionPoint.x; float aY = topCollisionPoint.y; float bX = bottomCollisionPoint.x; float bY = bottomCollisionPoint.y; // Calculate the midpoint between the top and bottom collision points. Vector2 verticalMidpoint = new Vector2((aX + bX) * 0.5f, (aY + bY) * 0.5f); if (TryFixMaximumRectangle( geometryEdges, verticalMidpoint, angleRadians, Width * Height, out newWidth, out newHeight)) { Center = verticalMidpoint; Angle = FitAngles[angleIndex]; Width = newWidth; Height = newHeight; } } if (EdgeUtilities.IsValidPoint(leftCollisionPoint) && EdgeUtilities.IsValidPoint(rightCollisionPoint)) { float aX = leftCollisionPoint.x; float aY = leftCollisionPoint.y; float bX = rightCollisionPoint.x; float bY = rightCollisionPoint.y; // Calculate the midpoint between the left and right collision points. Vector2 horizontalMidpoint = new Vector2((aX + bX) * 0.5f, (aY + bY) * 0.5f); if (TryFixMaximumRectangle( geometryEdges, horizontalMidpoint, angleRadians, Width * Height, out newWidth, out newHeight)) { Center = horizontalMidpoint; Angle = FitAngles[angleIndex]; Width = newWidth; Height = newHeight; } } } } } /// /// Find points at which there are collisions with the geometry around a given point. /// /// The boundary geometry. /// The point around which collisions will be identified. /// The angle, in radians, at which the collision points will be oriented. /// Receives the coordinates of the upper collision point. /// Receives the coordinates of the lower collision point. /// Receives the coordinates of the left collision point. /// Receives the coordinates of the right collision point. /// /// True if all of the required collision points are located, false otherwise. /// If a point is unable to be found, the appropriate out parameter will be set to . /// private bool FindSurroundingCollisionPoints( Edge[] geometryEdges, Vector2 point, float angleRadians, out Vector2 topCollisionPoint, out Vector2 bottomCollisionPoint, out Vector2 leftCollisionPoint, out Vector2 rightCollisionPoint) { // Initialize out parameters. topCollisionPoint = EdgeUtilities.InvalidPoint; bottomCollisionPoint = EdgeUtilities.InvalidPoint; leftCollisionPoint = EdgeUtilities.InvalidPoint; rightCollisionPoint = EdgeUtilities.InvalidPoint; // Check to see if the point is inside the geometry. if (!EdgeUtilities.IsInsideBoundary(geometryEdges, point)) { return false; } // Define values that are outside of the maximum boundary size. float largeValue = EdgeUtilities.maxWidth; float smallValue = -largeValue; // Find the top and bottom collision points by creating a large line segment that goes through the point to MAX and MIN values on Y Vector2 topEndpoint = new Vector2(point.x, largeValue); Vector2 bottomEndpoint = new Vector2(point.x, smallValue); topEndpoint = RotatePoint(topEndpoint, point, angleRadians); bottomEndpoint = RotatePoint(bottomEndpoint, point, angleRadians); Edge verticalLine = new Edge(topEndpoint, bottomEndpoint); // Find the left and right collision points by creating a large line segment that goes through the point to MAX and Min values on X Vector2 rightEndpoint = new Vector2(largeValue, point.y); Vector2 leftEndpoint = new Vector2(smallValue, point.y); rightEndpoint = RotatePoint(rightEndpoint, point, angleRadians); leftEndpoint = RotatePoint(leftEndpoint, point, angleRadians); Edge horizontalLine = new Edge(rightEndpoint, leftEndpoint); for (int i = 0; i < geometryEdges.Length; i++) { // Look for a vertical collision Vector2 verticalIntersectionPoint = EdgeUtilities.GetIntersectionPoint(geometryEdges[i], verticalLine); if (EdgeUtilities.IsValidPoint(verticalIntersectionPoint)) { // Is the intersection above or below the point? if (RotatePoint(verticalIntersectionPoint, point, -angleRadians).y > point.y) { // Update the top collision point if (!EdgeUtilities.IsValidPoint(topCollisionPoint) || (Vector2.SqrMagnitude(point - verticalIntersectionPoint) < Vector2.SqrMagnitude(point - topCollisionPoint))) { topCollisionPoint = verticalIntersectionPoint; } } else { // Update the bottom collision point if (!EdgeUtilities.IsValidPoint(bottomCollisionPoint) || (Vector2.SqrMagnitude(point - verticalIntersectionPoint) < Vector2.SqrMagnitude(point - bottomCollisionPoint))) { bottomCollisionPoint = verticalIntersectionPoint; } } } // Look for a horizontal collision Vector2 horizontalIntersection = EdgeUtilities.GetIntersectionPoint(geometryEdges[i], horizontalLine); if (EdgeUtilities.IsValidPoint(horizontalIntersection)) { // Is this intersection to the left or the right of the point? if (RotatePoint(horizontalIntersection, point, -angleRadians).x < point.x) { // Update the left collision point if (!EdgeUtilities.IsValidPoint(leftCollisionPoint) || (Vector2.SqrMagnitude(point - horizontalIntersection) < Vector2.SqrMagnitude(point - leftCollisionPoint))) { leftCollisionPoint = horizontalIntersection; } } else { // Update the right collision point if (!EdgeUtilities.IsValidPoint(rightCollisionPoint) || (Vector2.SqrMagnitude(point - horizontalIntersection) < Vector2.SqrMagnitude(point - rightCollisionPoint))) { rightCollisionPoint = horizontalIntersection; } } } } // Each corner of the rectangle must intersect with the geometry. if (!EdgeUtilities.IsValidPoint(topCollisionPoint) || !EdgeUtilities.IsValidPoint(bottomCollisionPoint) || !EdgeUtilities.IsValidPoint(leftCollisionPoint) || !EdgeUtilities.IsValidPoint(rightCollisionPoint)) { return false; } return true; } /// /// Determine of the provided point lies within the defined rectangle. /// /// The point to check /// /// True if the point is within the rectangle's bounds, false otherwise. /// /// The rectangle is not valid. public bool IsInsideBoundary(Vector2 point) { if (!IsValid) { throw new InvalidOperationException("A point cannot be within an invalid rectangle."); } point -= Center; point = RotatePoint(point, Vector2.zero, MathUtilities.DegreesToRadians(-Angle)); bool inWidth = Mathf.Abs(point.x) <= (Width * 0.5f); bool inHeight = Mathf.Abs(point.y) <= (Height * 0.5f); return (inWidth && inHeight); } /// /// Rotate a two dimensional point about another point by the specified angle. /// /// The point to be rotated. /// The point about which the rotation is to occur. /// The angle for the rotation, in radians /// /// The coordinates of the rotated point. /// private Vector2 RotatePoint(Vector2 point, Vector2 origin, float angleRadians) { if (angleRadians.Equals(0f)) { return point; } Vector2 rotated = point; // Translate to origin of rotation rotated.x -= origin.x; rotated.y -= origin.y; // Rotate the point float sin = Mathf.Sin(angleRadians); float cos = Mathf.Cos(angleRadians); float x = rotated.x * cos - rotated.y * sin; float y = rotated.x * sin + rotated.y * cos; // Translate back and return rotated.x = x + origin.x; rotated.y = y + origin.y; return rotated; } /// /// Check to see if a rectangle centered at the specified point and oriented at /// the specified angle will fit within the geometry. /// /// The boundary geometry. /// The center point of the rectangle. /// The orientation, in radians, of the rectangle. /// The width of the rectangle. /// The height of the rectangle. private bool CheckRectangleFit( Edge[] geometryEdges, Vector2 centerPoint, float angleRadians, float width, float height) { float halfWidth = width * 0.5f; float halfHeight = height * 0.5f; // Calculate the rectangle corners. Vector2 topLeft = new Vector2(centerPoint.x - halfWidth, centerPoint.y + halfHeight); Vector2 topRight = new Vector2(centerPoint.x + halfWidth, centerPoint.y + halfHeight); Vector2 bottomLeft = new Vector2(centerPoint.x - halfWidth, centerPoint.y - halfHeight); Vector2 bottomRight = new Vector2(centerPoint.x + halfWidth, centerPoint.y - halfHeight); // Rotate the rectangle. topLeft = RotatePoint(topLeft, centerPoint, angleRadians); topRight = RotatePoint(topRight, centerPoint, angleRadians); bottomLeft = RotatePoint(bottomLeft, centerPoint, angleRadians); bottomRight = RotatePoint(bottomRight, centerPoint, angleRadians); // Get the rectangle edges. Edge topEdge = new Edge(topLeft, topRight); Edge rightEdge = new Edge(topRight, bottomRight); Edge bottomEdge = new Edge(bottomLeft, bottomRight); Edge leftEdge = new Edge(topLeft, bottomLeft); // Check for collisions with the boundary geometry. If any of our edges collide, // the rectangle will not fit within the playspace. for (int i = 0; i < geometryEdges.Length; i++) { if (EdgeUtilities.IsValidPoint(EdgeUtilities.GetIntersectionPoint(geometryEdges[i], topEdge)) || EdgeUtilities.IsValidPoint(EdgeUtilities.GetIntersectionPoint(geometryEdges[i], rightEdge)) || EdgeUtilities.IsValidPoint(EdgeUtilities.GetIntersectionPoint(geometryEdges[i], bottomEdge)) || EdgeUtilities.IsValidPoint(EdgeUtilities.GetIntersectionPoint(geometryEdges[i], leftEdge))) { return false; } } // No collisions found with the rectangle. Success! return true; } /// /// Attempt to fit the largest rectangle possible within the geometry. /// /// The boundary geometry. /// The center point for the rectangle. /// The rotation, in radians, of the rectangle. /// The smallest allowed area. /// Returns the width of the rectangle. /// Returns the height of the rectangle. /// /// True if a rectangle with an area greater than or equal to minArea was able to be fit /// within the geometry at centerPoint. /// private bool TryFixMaximumRectangle( Edge[] geometryEdges, Vector2 centerPoint, float angleRadians, float minArea, out float width, out float height) { width = 0.0f; height = 0.0f; Vector2 topCollisionPoint; Vector2 bottomCollisionPoint; Vector2 leftCollisionPoint; Vector2 rightCollisionPoint; // Find the collision points with the geometry if (!FindSurroundingCollisionPoints(geometryEdges, centerPoint, angleRadians, out topCollisionPoint, out bottomCollisionPoint, out leftCollisionPoint, out rightCollisionPoint)) { return false; } // Start by calculating max width and height by ray-casting a cross from the point at the given angle // and taking the shortest leg of each ray. Width is the longest. float verticalMinDistanceToEdge = Mathf.Min( Vector2.Distance(centerPoint, topCollisionPoint), Vector2.Distance(centerPoint, bottomCollisionPoint)); float horizontalMinDistanceToEdge = Mathf.Min( Vector2.Distance(centerPoint, leftCollisionPoint), Vector2.Distance(centerPoint, rightCollisionPoint)); // Width is the largest of the possible dimensions float maxWidth = Math.Max(verticalMinDistanceToEdge, horizontalMinDistanceToEdge) * 2.0f; float maxHeight = Math.Min(verticalMinDistanceToEdge, horizontalMinDistanceToEdge) * 2.0f; float aspectRatio = 0.0f; // For each aspect ratio we do a binary search to find the maximum rectangle that fits, // though once we start increasing our area by minimumHeightGain we call it good enough. for (int i = 0; i < AspectRatios.Length; i++) { // The height is limited by the width. If a height would make our width exceed maxWidth, it can't be used float searchHeightUpperBound = Mathf.Max(maxHeight, maxWidth / AspectRatios[i]); // Set to the min height that will out perform our previous area at the given aspect ratio. This is 0 the first time. // Derived from biggestAreaSoFar=height*(height*aspectRatio) float searchHeightLowerBound = Mathf.Sqrt(Mathf.Max((width * height), minArea) / AspectRatios[i]); // If the lowest value needed to outperform the previous best is greater than our max, // this aspect ratio can't outperform what we've already calculated. if ((searchHeightLowerBound > searchHeightUpperBound) || (searchHeightLowerBound * AspectRatios[i] > maxWidth)) { continue; } float currentTestingHeight = Mathf.Max(searchHeightLowerBound, maxHeight * 0.5f); // Perform the binary search until continuing to search will not give us a significant win. do { if (CheckRectangleFit(geometryEdges, centerPoint, angleRadians, AspectRatios[i] * currentTestingHeight, currentTestingHeight)) { // Binary search up-ward // If the rectangle will fit, increase the lower bounds of our binary search searchHeightLowerBound = currentTestingHeight; width = currentTestingHeight * AspectRatios[i]; height = currentTestingHeight; aspectRatio = AspectRatios[i]; currentTestingHeight = (searchHeightUpperBound + currentTestingHeight) * 0.5f; } else { // If the rectangle won't fit, update our upper bound and lower our binary search searchHeightUpperBound = currentTestingHeight; currentTestingHeight = (currentTestingHeight + searchHeightLowerBound) * 0.5f; } } while ((searchHeightUpperBound - searchHeightLowerBound) > minimumHeightGain); } return (aspectRatio > 0.0f); } } }