diff --git a/osu.Framework.Tests/MathUtils/TestPathApproximator.cs b/osu.Framework.Tests/MathUtils/TestPathApproximator.cs index f545f42a29..8b32b06d78 100644 --- a/osu.Framework.Tests/MathUtils/TestPathApproximator.cs +++ b/osu.Framework.Tests/MathUtils/TestPathApproximator.cs @@ -1,7 +1,9 @@ // Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. +using System; using System.Collections.Generic; +using System.Linq; using NUnit.Framework; using osu.Framework.Utils; using osuTK; @@ -40,5 +42,24 @@ public void TestBSpline() Assert.True(Precision.AlmostEquals(approximated[28], points[6], 1e-4f)); Assert.True(Precision.AlmostEquals(approximated[10], new Vector2(-0.11415f, -0.69065f), 1e-4f)); } + + [Test] + public void TestBSplineThrowsOnInvalidDegree() + { + Vector2[] points = { new Vector2(0, 0), new Vector2(1, 0), new Vector2(1, -1), new Vector2(-1, -1), new Vector2(-1, 1), new Vector2(3, 2), new Vector2(3, 0) }; + + Assert.Throws(() => PathApproximator.BSplineToPiecewiseLinear(points, 0)); + Assert.Throws(() => PathApproximator.BSplineToPiecewiseLinear(points, -5)); + } + + [TestCase(0)] + [TestCase(1)] + public void TestBSplineDoesNothingWhenGivenTooFewPoints(int pointCount) + { + var points = Enumerable.Repeat(new Vector2(), pointCount).ToArray(); + + Assert.That(PathApproximator.BSplineToPiecewiseLinear(points, 3), Is.EquivalentTo(points)); + Assert.That(PathApproximator.BezierToPiecewiseLinear(points), Is.EquivalentTo(points)); + } } } diff --git a/osu.Framework.Tests/Visual/Drawables/TestSceneInteractivePathDrawing.cs b/osu.Framework.Tests/Visual/Drawables/TestSceneInteractivePathDrawing.cs index 4ebe43eca6..df0f3a2647 100644 --- a/osu.Framework.Tests/Visual/Drawables/TestSceneInteractivePathDrawing.cs +++ b/osu.Framework.Tests/Visual/Drawables/TestSceneInteractivePathDrawing.cs @@ -59,17 +59,21 @@ public TestSceneInteractivePathDrawing() { bSplineBuilder.Degree = v; }); - AddSliderStep($"{nameof(bSplineBuilder.Tolerance)}", 0f, 1f, 0.1f, v => + AddSliderStep($"{nameof(bSplineBuilder.Tolerance)}", 0f, 3f, 1.5f, v => { bSplineBuilder.Tolerance = v; }); + AddSliderStep($"{nameof(bSplineBuilder.CornerThreshold)}", 0f, 1f, 0.4f, v => + { + bSplineBuilder.CornerThreshold = v; + }); } private void updateControlPointsViz() { controlPointViz.Clear(); - foreach (var cp in bSplineBuilder.GetControlPoints()) + foreach (var cp in bSplineBuilder.ControlPoints) { controlPointViz.Add(new Box { diff --git a/osu.Framework/Utils/IncrementalBSplineBuilder.cs b/osu.Framework/Utils/IncrementalBSplineBuilder.cs index 1f4b9eef39..fbab87371b 100644 --- a/osu.Framework/Utils/IncrementalBSplineBuilder.cs +++ b/osu.Framework/Utils/IncrementalBSplineBuilder.cs @@ -6,6 +6,7 @@ using System.Diagnostics; using System.Linq; using osu.Framework.Caching; +using osu.Framework.Graphics.Primitives; using osuTK; namespace osu.Framework.Utils @@ -18,13 +19,66 @@ namespace osu.Framework.Utils public class IncrementalBSplineBuilder { private readonly List inputPath = new List(); + private readonly List cumulativeInputPathLength = new List(); + + private static Vector2 getPathAt(List path, List cumulativeDistances, float t) + { + if (path.Count == 0) + throw new InvalidOperationException("Input path is empty."); + else if (path.Count == 1) + return path[0]; + + if (t <= 0) + return path[0]; + + if (t >= cumulativeDistances[^1]) + return path[^1]; + + int index = cumulativeDistances.BinarySearch(t); + if (index < 0) + index = ~index; + + float lengthBefore = index == 0 ? 0 : cumulativeDistances[index - 1]; + float lengthAfter = cumulativeDistances[index]; + float segmentLength = lengthAfter - lengthBefore; + float segmentT = (t - lengthBefore) / segmentLength; + + return Vector2.Lerp(path[index], path[index + 1], segmentT); + } + + private float inputPathLength => cumulativeInputPathLength.Count == 0 ? 0 : cumulativeInputPathLength[^1]; + + /// + /// Spacing to use in spline-related finite difference (FD) calculations. + /// + internal const float FD_EPSILON = PathApproximator.BEZIER_TOLERANCE * 8f; + + /// + /// Get the absolute amount of rotation (in radians) at a given point on the path. + /// + /// The path to get the rotation from. + /// The cumulative distances of the path. + /// The point on the path to get the rotation from. + /// The absolute amount of rotation (in radians) at the given point on the path. + private static float getAbsWindingAt(List path, List cumulativeDistances, float t) + { + Vector2 xminus = getPathAt(path, cumulativeDistances, t - FD_EPSILON); + Vector2 x = getPathAt(path, cumulativeDistances, t); + Vector2 xplus = getPathAt(path, cumulativeDistances, t + FD_EPSILON); + Vector2 tminus = x == xminus ? Vector2.Zero : (x - xminus).Normalized(); + Vector2 tplus = xplus == x ? Vector2.Zero : (xplus - x).Normalized(); + return MathF.Abs(MathF.Acos(Math.Clamp(Vector2.Dot(tminus, tplus), -1f, 1f))); + } private readonly Cached> outputCache = new Cached> { Value = new List() }; - private readonly List controlPoints = new List(); + private readonly Cached> controlPoints = new Cached> + { + Value = new List() + }; private int degree; @@ -42,13 +96,14 @@ public int Degree degree = value; outputCache.Invalidate(); + controlPoints.Invalidate(); } } private float tolerance; /// - /// Gets or sets the tolerance for determining when to add a new control point. Must not be negative. Default is 0.1. + /// Gets or sets the tolerance for determining when to add a new control point. Must not be negative. Default is 1.5. /// /// Thrown when the value is negative. public float Tolerance @@ -61,6 +116,27 @@ public float Tolerance tolerance = value; outputCache.Invalidate(); + controlPoints.Invalidate(); + } + } + + private float cornerThreshold; + + /// + /// Gets or sets the corner threshold for determining when to add a new control point. Must not be negative. Default is 0.4. + /// + /// Thrown when the value is negative. + public float CornerThreshold + { + get => cornerThreshold; + set + { + if (cornerThreshold < 0) + throw new ArgumentOutOfRangeException(nameof(value), "CornerThreshold must not be negative."); + + cornerThreshold = value; + outputCache.Invalidate(); + controlPoints.Invalidate(); } } @@ -78,43 +154,211 @@ public IReadOnlyList OutputPath } } + /// + /// The list of control points of the B-Spline. This is inferred from the input path. + /// + public IReadOnlyList ControlPoints + { + get + { + if (!controlPoints.IsValid) + regenerateApproximatedPathControlPoints(); + + return controlPoints.Value; + } + } + /// /// Initializes a new instance of the class with specified degree and tolerance. /// /// The degree of the B-Spline. /// The tolerance for control point addition. - public IncrementalBSplineBuilder(int degree = 3, float tolerance = 0.1f) + /// The threshold to use for inserting sharp control points at corners. + public IncrementalBSplineBuilder(int degree = 3, float tolerance = 1.5f, float cornerThreshold = 0.4f) { Degree = degree; Tolerance = tolerance; + CornerThreshold = cornerThreshold; } - /// - /// The list of control points of the B-Spline. This is inferred from the input path. - /// - public IReadOnlyList GetControlPoints() - => controlPoints.ToArray(); - /// /// The list of input points. /// public IReadOnlyList GetInputPath() => inputPath.ToArray(); - private void redrawApproximatedPath() + /// + /// Computes a smoothed version of the input path by generating a high-degree BSpline from densely + /// spaces samples of the input path. + /// + /// A tuple containing the smoothed vertices and the cumulative distances of the smoothed path. + private (List vertices, List distances) computeSmoothedInputPath() { - // Set value of output cache to update the cache to be valid. - outputCache.Value = new List(); - if (inputPath.Count == 0) + var cps = new Vector2[(int)(inputPathLength / FD_EPSILON)]; + for (int i = 0; i < cps.Length; ++i) + cps[i] = getPathAt(inputPath, cumulativeInputPathLength, i * FD_EPSILON); + + // Empirically, degree 7 works really well as a good tradeoff for smoothing vs sharpness here. + const int smoothed_input_path_degree = 7; + var vertices = PathApproximator.BSplineToPiecewiseLinear(cps, smoothed_input_path_degree); + var distances = new List(); + float cumulativeLength = 0; + + for (int i = 1; i < vertices.Count; ++i) + { + cumulativeLength += Vector2.Distance(vertices[i], vertices[i - 1]); + distances.Add(cumulativeLength); + } + + return (vertices, distances); + } + + /// + /// Detects corners in the input path by thresholding how much the path curves and checking + /// whether this curvature is local (i.e. a corner rather than a smooth, yet tight turn). + /// + /// The vertices of the input path. + /// The cumulative distances of the input path. + /// A list of t values at which corners occur. + private List detectCorners(List vertices, List distances) + { + var cornerT = new List { 0f }; + + float threshold = cornerThreshold / FD_EPSILON; + + const float step_size = FD_EPSILON; + int nSteps = (int)(distances[^1] / step_size); + + // Empirically, averaging the winding rate over a neighborhood of 32 samples seems to be + // a good representation of the neighborhood of the curve. + const int n_avg_samples = 32; + float avgCurvature = 0.0f; + + for (int i = 0; i < nSteps; ++i) + { + // Update average curvature by adding the new winding rate and subtracting the old one from + // nAvgSamples steps ago. + float newt = i * step_size; + float newWinding = getAbsWindingAt(vertices, distances, newt); + + float oldt = (i - n_avg_samples) * step_size; + float oldWinding = oldt < 0 ? 0 : getAbsWindingAt(vertices, distances, oldt); + + avgCurvature += (newWinding - oldWinding) / n_avg_samples; + + // Check whether the current winding rate is a local maximum and whether it exceeds the + // threshold as well as the surrounding average curvature. If so, we have found a corner. + // Also prohibit marking new corners that are too close to the previous one, where "too close" + // is defined as the averaging windows overlapping. This ensures the same corner can not + // be detected twice. + float midt = (i - n_avg_samples / 2f) * step_size; + float midWinding = midt < 0 ? 0 : getAbsWindingAt(vertices, distances, midt); + + float distToPrevCorner = cornerT.Count == 0 ? float.MaxValue : newt - cornerT[^1]; + if (midWinding > threshold && midWinding > avgCurvature * 4 && distToPrevCorner > n_avg_samples * step_size) + cornerT.Add(midt); + } + + // The end of the path is by definition a corner + cornerT.Add(distances[^1]); + return cornerT; + } + + private void regenerateApproximatedPathControlPoints() + { + // Approximating a given input path with a BSpline has three stages: + // 1. Fit a dense-ish BSpline (with one control point in FdEpsilon-sized intervals) to the input path. + // The purpose of this dense BSpline is an initial smoothening that permits reliable curvature + // analysis in the next steps. + // 2. Detect corners by thresholding local curvature maxima and place sharp control points at these corners. + // 3. Place additional control points inbetween the sharp ones with density proportional to the product + // of Tolerance and curvature. + // 4. Additionally, we special case linear segments: if the path does not deviate more + // than some threshold from a straight line, we do not add additional control points. + var (vertices, distances) = computeSmoothedInputPath(); + + if (vertices.Count < 2) + { + controlPoints.Value = vertices; return; + } - var oldInputs = inputPath.ToList(); + controlPoints.Value = new List(); - inputPath.Clear(); - controlPoints.Clear(); + Debug.Assert(vertices.Count == distances.Count + 1); + var cornerTs = detectCorners(vertices, distances); + + var cps = controlPoints.Value; + cps.Add(vertices[0]); + + // Populate each segment between corners with control points that have density proportional to the + // product of Tolerance and curvature. + const float step_size = FD_EPSILON; + + for (int i = 1; i < cornerTs.Count; ++i) + { + float totalWinding = 0; + + float t0 = cornerTs[i - 1] + step_size * 2; + float t1 = cornerTs[i] - step_size * 2; + + Vector2 c0 = getPathAt(vertices, distances, cornerTs[i - 1]); + Vector2 c1 = getPathAt(vertices, distances, cornerTs[i]); + Line linearConnection = new Line(c0, c1); + + var tmp = new List(); + bool allOnLine = true; + float onLineThreshold = 5 * Tolerance * step_size; + + if (t1 > t0) + { + int nSteps = (int)((t1 - t0) / step_size); + + for (int j = 0; j < nSteps; ++j) + { + float t = t0 + j * step_size; + totalWinding += getAbsWindingAt(vertices, distances, t); + } + + int nControlPoints = (int)(totalWinding / Tolerance); + float controlPointSpacing = totalWinding / nControlPoints; + float currentWinding = 0; + + for (int j = 0; j < nSteps; ++j) + { + float t = t0 + j * step_size; + + if (currentWinding > controlPointSpacing) + { + Vector2 p = getPathAt(vertices, distances, t); + if (linearConnection.DistanceSquaredToPoint(p) > onLineThreshold * onLineThreshold) + allOnLine = false; + + tmp.Add(p); + currentWinding -= controlPointSpacing; + } + + currentWinding += getAbsWindingAt(vertices, distances, t); + } + } + + if (!allOnLine) + cps.AddRange(tmp); + + // Insert the corner at the end of the segment as a sharp control point consisting of + // degree many regular control points, meaning that the BSpline will have a kink here. + // Special case the last corner which will be the end of the path and thus automatically + // duplicated degree times by BSplineToPiecewiseLinear down the line. + if (i == cornerTs.Count - 1) + cps.Add(c1); + else + cps.AddRange(Enumerable.Repeat(c1, degree)); + } + } - foreach (var v in oldInputs) - AddLinearPoint(v); + private void redrawApproximatedPath() + { + outputCache.Value = PathApproximator.BSplineToPiecewiseLinear(ControlPoints.ToArray(), degree); } /// @@ -123,7 +367,9 @@ private void redrawApproximatedPath() public void Clear() { inputPath.Clear(); - controlPoints.Clear(); + cumulativeInputPathLength.Clear(); + + controlPoints.Value = new List(); outputCache.Value = new List(); } @@ -133,61 +379,21 @@ public void Clear() /// The vector representing the point to add. public void AddLinearPoint(Vector2 v) { - if (!outputCache.IsValid) - redrawApproximatedPath(); + outputCache.Invalidate(); + controlPoints.Invalidate(); if (inputPath.Count == 0) { inputPath.Add(v); - // We add the first point twice so we can track the - // end point of the raw path using the last control point. - controlPoints.Add(inputPath[0]); - controlPoints.Add(inputPath[0]); return; } - inputPath.Add(v); - - var cps = controlPoints; - Debug.Assert(cps.Count >= 2); - - cps[^1] = inputPath[^1]; - - // Calculate the normalized momentum vectors for both raw and approximated paths. - // Momentum here refers to a direction vector representing the path's direction of movement. - var mraw = momentumDirection(inputPath, 3); - var mcp = cps.Count > 2 ? momentumDirection(outputCache.Value, 1) : Vector2.Zero; - - // Determine the alignment between the raw path and control path's momentums. - // It uses Vector2.Dot which calculates the cosine of the angle between two vectors. - // This alignment is used to adjust the control points based on the path's direction change. - float alignment = mraw == Vector2.Zero || mcp == Vector2.Zero ? 1.0f : MathF.Max(Vector2.Dot(mraw, mcp), 0.01f); - - // Calculate the distance between the last two control points. - // This distance is then used, along with alignment, to decide if a new control point is needed. - // The threshold for adding a new control point is based on the alignment and a predefined accuracy factor. - float distance = Vector2.Distance(cps[^1], cps[^2]); - if (distance / MathF.Pow(alignment, 4) > Tolerance * 1000) - cps.Add(cps[^1]); - - outputCache.Value = PathApproximator.BSplineToPiecewiseLinear(cps.ToArray(), degree); - } - - private Vector2 momentumDirection(IReadOnlyList vertices, int window) - { - if (vertices.Count < window + 1) - return Vector2.Zero; - - var sum = Vector2.Zero; - for (int i = 0; i < window; i++) - sum += vertices[^(i + 1)] - vertices[^(i + 2)]; - - // We choose a very small acceptable difference here to ensure that - // small momenta are not ignored. - if (Precision.AlmostEquals(sum.X, 0, 1e-7f) && Precision.AlmostEquals(sum.Y, 0, 1e-7f)) - return Vector2.Zero; + float inputDistance = Vector2.Distance(v, inputPath[^1]); + if (inputDistance < FD_EPSILON * 2) + return; - return (sum / window).Normalized(); + inputPath.Add(v); + cumulativeInputPathLength.Add((cumulativeInputPathLength.Count == 0 ? 0 : cumulativeInputPathLength[^1]) + inputDistance); } } } diff --git a/osu.Framework/Utils/PathApproximator.cs b/osu.Framework/Utils/PathApproximator.cs index 925d663c98..887572007a 100644 --- a/osu.Framework/Utils/PathApproximator.cs +++ b/osu.Framework/Utils/PathApproximator.cs @@ -14,7 +14,7 @@ namespace osu.Framework.Utils /// public static class PathApproximator { - private const float bezier_tolerance = 0.25f; + internal const float BEZIER_TOLERANCE = 0.25f; /// /// The amount of pieces to calculate for each control point quadruplet. @@ -31,7 +31,7 @@ public static class PathApproximator /// A list of vectors representing the piecewise-linear approximation. public static List BezierToPiecewiseLinear(ReadOnlySpan controlPoints) { - return BSplineToPiecewiseLinear(controlPoints, controlPoints.Length - 1); + return BSplineToPiecewiseLinear(controlPoints, Math.Max(1, controlPoints.Length - 1)); } /// @@ -317,7 +317,7 @@ private static bool bezierIsFlatEnough(Vector2[] controlPoints) { for (int i = 1; i < controlPoints.Length - 1; i++) { - if ((controlPoints[i - 1] - 2 * controlPoints[i] + controlPoints[i + 1]).LengthSquared > bezier_tolerance * bezier_tolerance * 4) + if ((controlPoints[i - 1] - 2 * controlPoints[i] + controlPoints[i + 1]).LengthSquared > BEZIER_TOLERANCE * BEZIER_TOLERANCE * 4) return false; }