Skip to content

Commit

Permalink
Merge pull request #6044 from default0/path-approximation
Browse files Browse the repository at this point in the history
Add class for efficiently building B-Splines
  • Loading branch information
bdach authored Nov 13, 2023
2 parents 5c5c6c5 + aefd351 commit 1962cf6
Show file tree
Hide file tree
Showing 8 changed files with 357 additions and 45 deletions.
4 changes: 2 additions & 2 deletions osu.Framework.Tests/MathUtils/TestPathApproximator.cs
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ public void TestLagrange()
// lagrange of (0,0) (0.5,0.35) (1,1) is equal to 0.6x*x + 0.4x
Vector2[] points = { new Vector2(0, 0), new Vector2(0.5f, 0.35f), new Vector2(1, 1) };

List<Vector2> approximated = PathApproximator.ApproximateLagrangePolynomial(points);
List<Vector2> approximated = PathApproximator.LagrangePolynomialToPiecewiseLinear(points);
Assert.Greater(approximated.Count, 10, "Approximated polynomial should have at least 10 points to test");

for (int i = 0; i < approximated.Count; i++)
Expand All @@ -34,7 +34,7 @@ public void TestBSpline()
{
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) };

List<Vector2> approximated = PathApproximator.ApproximateBSpline(points, 4);
List<Vector2> approximated = PathApproximator.BSplineToPiecewiseLinear(points, 4);
Assert.AreEqual(approximated.Count, 29, "Approximated path should have 29 points to test");
Assert.True(Precision.AlmostEquals(approximated[0], points[0], 1e-4f));
Assert.True(Precision.AlmostEquals(approximated[28], points[6], 1e-4f));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ protected override void LoadComplete()
if (copy.Length != 3)
return;

path.Vertices = PathApproximator.ApproximateCircularArc(copy);
path.Vertices = PathApproximator.CircularArcToPiecewiseLinear(copy);

var bounds = PathApproximator.CircularArcBoundingBox(copy);
boundingBox.Size = bounds.Size;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -165,7 +165,7 @@ protected override void Update()
vectorPath.AddRange(ordered.Select(p => p.PointPosition.Value));
vectorPath.Add(new Vector2(DrawWidth, 0));

var bezierPath = PathApproximator.ApproximateBezier(vectorPath.ToArray());
var bezierPath = PathApproximator.BezierToPiecewiseLinear(vectorPath.ToArray());
path.Vertices = bezierPath;
path.Position = -path.PositionInBoundingBox(Vector2.Zero);

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,109 @@
// Copyright (c) ppy Pty Ltd <[email protected]>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.

using osu.Framework.Graphics;
using osuTK.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Graphics.Lines;
using osu.Framework.Input.Events;
using osuTK.Input;
using osu.Framework.Utils;
using osuTK;

namespace osu.Framework.Tests.Visual.Drawables
{
[System.ComponentModel.Description("Approximate a hand-drawn path with minimal B-spline control points")]
public partial class TestSceneInteractivePathDrawing : FrameworkTestScene
{
private readonly Path rawDrawnPath;
private readonly Path approximatedDrawnPath;
private readonly Container controlPointViz;

private readonly IncrementalBSplineBuilder bSplineBuilder = new IncrementalBSplineBuilder();

public TestSceneInteractivePathDrawing()
{
Child = new Container
{
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
rawDrawnPath = new Path
{
Colour = Color4.DeepPink,
PathRadius = 5,
},
approximatedDrawnPath = new Path
{
Colour = Color4.Blue,
PathRadius = 3,
},
controlPointViz = new Container
{
RelativeSizeAxes = Axes.Both,
Alpha = 0.5f,
},
}
};

updateViz();
OnUpdate += _ => updateViz();

AddStep("Reset path", () =>
{
bSplineBuilder.Clear();
});

AddSliderStep($"{nameof(bSplineBuilder.Degree)}", 1, 5, 3, v =>
{
bSplineBuilder.Degree = v;
});
AddSliderStep($"{nameof(bSplineBuilder.Tolerance)}", 0f, 1f, 0.1f, v =>
{
bSplineBuilder.Tolerance = v;
});
}

private void updateControlPointsViz()
{
controlPointViz.Clear();

foreach (var cp in bSplineBuilder.GetControlPoints())
{
controlPointViz.Add(new Box
{
Origin = Anchor.Centre,
Size = new Vector2(10),
Position = cp,
Colour = Color4.LightGreen,
});
}
}

protected override bool OnDragStart(DragStartEvent e)
{
if (e.Button == MouseButton.Left)
{
bSplineBuilder.Clear();
bSplineBuilder.AddLinearPoint(rawDrawnPath.ToLocalSpace(ToScreenSpace(e.MousePosition)));
return true;
}

return false;
}

private void updateViz()
{
rawDrawnPath.Vertices = bSplineBuilder.GetInputPath();
approximatedDrawnPath.Vertices = bSplineBuilder.OutputPath;

updateControlPointsViz();
}

protected override void OnDrag(DragEvent e)
{
bSplineBuilder.AddLinearPoint(rawDrawnPath.ToLocalSpace(ToScreenSpace(e.MousePosition)));
}
}
}
20 changes: 10 additions & 10 deletions osu.Framework.Tests/Visual/Drawables/TestScenePathApproximator.cs
Original file line number Diff line number Diff line change
Expand Up @@ -20,26 +20,26 @@ public TestScenePathApproximator()
{
Cell(0).AddRange(new[]
{
createLabel("ApproximateBezier"),
new ApproximatedPathTest(PathApproximator.ApproximateBezier),
createLabel(nameof(PathApproximator.BezierToPiecewiseLinear)),
new ApproximatedPathTest(PathApproximator.BezierToPiecewiseLinear),
});

Cell(1).AddRange(new[]
{
createLabel("ApproximateCatmull"),
new ApproximatedPathTest(PathApproximator.ApproximateCatmull),
createLabel(nameof(PathApproximator.CatmullToPiecewiseLinear)),
new ApproximatedPathTest(PathApproximator.CatmullToPiecewiseLinear),
});

Cell(2).AddRange(new[]
{
createLabel("ApproximateCircularArc"),
new ApproximatedPathTest(PathApproximator.ApproximateCircularArc),
createLabel(nameof(PathApproximator.CircularArcToPiecewiseLinear)),
new ApproximatedPathTest(PathApproximator.CircularArcToPiecewiseLinear),
});

Cell(3).AddRange(new[]
{
createLabel("ApproximateLagrangePolynomial"),
new ApproximatedPathTest(PathApproximator.ApproximateLagrangePolynomial),
createLabel(nameof(PathApproximator.LagrangePolynomialToPiecewiseLinear)),
new ApproximatedPathTest(PathApproximator.LagrangePolynomialToPiecewiseLinear),
});
}

Expand All @@ -50,10 +50,10 @@ public TestScenePathApproximator()
Colour = Color4.White,
};

public delegate List<Vector2> ApproximatorFunc(ReadOnlySpan<Vector2> controlPoints);

private partial class ApproximatedPathTest : SmoothPath
{
public delegate List<Vector2> ApproximatorFunc(ReadOnlySpan<Vector2> controlPoints);

public ApproximatedPathTest(ApproximatorFunc approximator)
{
Vector2[] points = new Vector2[5];
Expand Down
10 changes: 10 additions & 0 deletions osu.Framework/Graphics/Lines/Path.cs
Original file line number Diff line number Diff line change
Expand Up @@ -237,6 +237,16 @@ public void AddVertex(Vector2 pos)
Invalidate(Invalidation.DrawSize);
}

public void ReplaceVertex(int index, Vector2 pos)
{
vertices[index] = pos;

vertexBoundsCache.Invalidate();
segmentsCache.Invalidate();

Invalidate(Invalidation.DrawSize);
}

private readonly List<Line> segmentsBacking = new List<Line>();
private readonly Cached segmentsCache = new Cached();
private List<Line> segments => segmentsCache.IsValid ? segmentsBacking : generateSegments();
Expand Down
Loading

0 comments on commit 1962cf6

Please sign in to comment.