using UnityEngine;
namespace Cinemachine.Utility
{
///
/// A collection of utilities relating to Bezier splines
///
public static class SplineHelpers
{
/// Compute the value of a 4-point 3-dimensional bezier spline
/// How far along the spline (0...1)
/// First point
/// First tangent
/// Second point
/// Second tangent
/// The spline value at t
public static Vector3 Bezier3(
float t, Vector3 p0, Vector3 p1, Vector3 p2, Vector3 p3)
{
t = Mathf.Clamp01(t);
float d = 1f - t;
return d * d * d * p0 + 3f * d * d * t * p1
+ 3f * d * t * t * p2 + t * t * t * p3;
}
/// Compute the tangent of a 4-point 3-dimensional bezier spline
/// How far along the spline (0...1)
/// First point
/// First tangent
/// Second point
/// Second tangent
/// The spline tangent at t
public static Vector3 BezierTangent3(
float t, Vector3 p0, Vector3 p1, Vector3 p2, Vector3 p3)
{
t = Mathf.Clamp01(t);
return (-3f * p0 + 9f * p1 - 9f * p2 + 3f * p3) * (t * t)
+ (6f * p0 - 12f * p1 + 6f * p2) * t
- 3f * p0 + 3f * p1;
}
/// Compute the weights for the tangent of a 4-point 3-dimensional bezier spline
/// First point
/// First tangent
/// Second point
/// Second tangent
/// First output weight
/// Second output weight
/// Third output weight
/// Weights for the bezier tangent
public static void BezierTangentWeights3(
Vector3 p0, Vector3 p1, Vector3 p2, Vector3 p3,
out Vector3 w0, out Vector3 w1, out Vector3 w2)
{
w0 = -3f * p0 + 9f * p1 - 9f * p2 + 3f * p3;
w1 = 6f * p0 - 12f * p1 + 6f * p2;
w2 = -3f * p0 + 3f * p1;
}
/// Compute the value of a 4-point 1-dimensional bezier spline
/// How far along the spline (0...1)
/// First point
/// First tangent
/// Second point
/// Second tangent
/// The spline value at t
public static float Bezier1(float t, float p0, float p1, float p2, float p3)
{
t = Mathf.Clamp01(t);
float d = 1f - t;
return d * d * d * p0 + 3f * d * d * t * p1
+ 3f * d * t * t * p2 + t * t * t * p3;
}
/// Compute the tangent of a 4-point 1-dimensional bezier spline
/// How far along the spline (0...1)
/// First point
/// First tangent
/// Second point
/// Second tangent
/// The spline tangent at t
public static float BezierTangent1(
float t, float p0, float p1, float p2, float p3)
{
t = Mathf.Clamp01(t);
return (-3f * p0 + 9f * p1 - 9f * p2 + 3f * p3) * t * t
+ (6f * p0 - 12f * p1 + 6f * p2) * t
- 3f * p0 + 3f * p1;
}
///
/// Use the Thomas algorithm to compute smooth tangent values for a spline.
/// Resultant tangents guarantee second-order smoothness of the curve.
///
/// The knots defining the curve
/// Output buffer for the first control points (1 per knot)
/// Output buffer for the second control points (1 per knot)
public static void ComputeSmoothControlPoints(
ref Vector4[] knot, ref Vector4[] ctrl1, ref Vector4[] ctrl2)
{
int numPoints = knot.Length;
if (numPoints <= 2)
{
if (numPoints == 2)
{
ctrl1[0] = Vector4.Lerp(knot[0], knot[1], 0.33333f);
ctrl2[0] = Vector4.Lerp(knot[0], knot[1], 0.66666f);
}
else if (numPoints == 1)
ctrl1[0] = ctrl2[0] = knot[0];
return;
}
var a = new float[numPoints];
var b = new float[numPoints];
var c = new float[numPoints];
var r = new float[numPoints];
for (int axis = 0; axis < 4; ++axis)
{
int n = numPoints - 1;
// Linear into the first segment
a[0] = 0;
b[0] = 2;
c[0] = 1;
r[0] = knot[0][axis] + 2 * knot[1][axis];
// Internal segments
for (int i = 1; i < n - 1; ++i)
{
a[i] = 1;
b[i] = 4;
c[i] = 1;
r[i] = 4 * knot[i][axis] + 2 * knot[i+1][axis];
}
// Linear out of the last segment
a[n - 1] = 2;
b[n - 1] = 7;
c[n - 1] = 0;
r[n - 1] = 8 * knot[n - 1][axis] + knot[n][axis];
// Solve with Thomas algorithm
for (int i = 1; i < n; ++i)
{
float m = a[i] / b[i-1];
b[i] = b[i] - m * c[i-1];
r[i] = r[i] - m * r[i-1];
}
// Compute ctrl1
ctrl1[n-1][axis] = r[n-1] / b[n-1];
for (int i = n - 2; i >= 0; --i)
ctrl1[i][axis] = (r[i] - c[i] * ctrl1[i + 1][axis]) / b[i];
// Compute ctrl2 from ctrl1
for (int i = 0; i < n; i++)
ctrl2[i][axis] = 2 * knot[i + 1][axis] - ctrl1[i + 1][axis];
ctrl2[n - 1][axis] = 0.5f * (knot[n][axis] + ctrl1[n - 1][axis]);
}
}
///
/// Use the Thomas algorithm to compute smooth tangent values for a spline.
/// This method will assume that the spline is looped (i.e. that the last knot is followed by the first).
/// Resultant tangents guarantee second-order smoothness of the curve.
///
/// The knots defining the curve
/// Output buffer for the first control points (1 per knot)
/// Output buffer for the second control points (1 per knot)
public static void ComputeSmoothControlPointsLooped(
ref Vector4[] knot, ref Vector4[] ctrl1, ref Vector4[] ctrl2)
{
int numPoints = knot.Length;
if (numPoints < 2)
{
if (numPoints == 1)
ctrl1[0] = ctrl2[0] = knot[0];
return;
}
int margin = Mathf.Min(4, numPoints-1);
Vector4[] knotLooped = new Vector4[numPoints + 2 * margin];
Vector4[] ctrl1Looped = new Vector4[numPoints + 2 * margin];
Vector4[] ctrl2Looped = new Vector4[numPoints + 2 * margin];
for (int i = 0; i < margin; ++i)
{
knotLooped[i] = knot[numPoints-(margin-i)];
knotLooped[numPoints+margin+i] = knot[i];
}
for (int i = 0; i < numPoints; ++i)
knotLooped[i + margin] = knot[i];
ComputeSmoothControlPoints(ref knotLooped, ref ctrl1Looped, ref ctrl2Looped);
for (int i = 0; i < numPoints; ++i)
{
ctrl1[i] = ctrl1Looped[i + margin];
ctrl2[i] = ctrl2Looped[i + margin];
}
}
}
}