You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
415 lines
12 KiB
JavaScript
415 lines
12 KiB
JavaScript
import { Cartesian3 } from "../../Source/Cesium.js";
|
|
import { HermiteSpline } from "../../Source/Cesium.js";
|
|
import { Math as CesiumMath } from "../../Source/Cesium.js";
|
|
|
|
describe("Core/HermiteSpline", function () {
|
|
var points;
|
|
var times;
|
|
|
|
beforeEach(function () {
|
|
points = [
|
|
new Cartesian3(-1.0, -1.0, 0.0),
|
|
new Cartesian3(-0.5, -0.125, 0.0),
|
|
new Cartesian3(0.5, 0.125, 0.0),
|
|
new Cartesian3(1.0, 1.0, 0.0),
|
|
];
|
|
times = [0.0, 1.0, 2.0, 3.0];
|
|
});
|
|
|
|
it("constructor throws without points, times or tangents", function () {
|
|
expect(function () {
|
|
return new HermiteSpline();
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("constructor throws when control points length is less than 2", function () {
|
|
expect(function () {
|
|
return new HermiteSpline({
|
|
points: [Cartesian3.ZERO],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("constructor throws when times.length is not equal to points.length", function () {
|
|
expect(function () {
|
|
return new HermiteSpline({
|
|
points: points,
|
|
times: [0.0, 1.0],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("constructor throws when inTangents or outTangents length is not equal to points.length - 1", function () {
|
|
expect(function () {
|
|
return new HermiteSpline({
|
|
points: points,
|
|
times: times,
|
|
inTangents: [Cartesian3.ZERO],
|
|
outTangents: [Cartesian3.UNIT_X],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
// returns a function for a hermite curve between points p and q
|
|
// with tangents pT and qT respectively on the interval [0, 1]
|
|
function createHermiteBasis(p, pT, q, qT) {
|
|
return function (u) {
|
|
var a = 2.0 * u * u * u - 3.0 * u * u + 1.0;
|
|
var b = -2.0 * u * u * u + 3.0 * u * u;
|
|
var c = u * u * u - 2.0 * u * u + u;
|
|
var d = u * u * u - u * u;
|
|
|
|
var p0 = Cartesian3.multiplyByScalar(p, a, new Cartesian3());
|
|
var p1 = Cartesian3.multiplyByScalar(q, b, new Cartesian3());
|
|
var p2 = Cartesian3.multiplyByScalar(pT, c, new Cartesian3());
|
|
var p3 = Cartesian3.multiplyByScalar(qT, d, new Cartesian3());
|
|
|
|
return Cartesian3.add(
|
|
Cartesian3.add(Cartesian3.add(p0, p1, p0), p2, p0),
|
|
p3,
|
|
p0
|
|
);
|
|
};
|
|
}
|
|
|
|
it("create spline", function () {
|
|
var hs = new HermiteSpline({
|
|
times: [0.0, 1.0, 3.0, 4.5, 6.0],
|
|
points: [
|
|
new Cartesian3(1235398.0, -4810983.0, 4146266.0),
|
|
new Cartesian3(1372574.0, -5345182.0, 4606657.0),
|
|
new Cartesian3(-757983.0, -5542796.0, 4514323.0),
|
|
new Cartesian3(-2821260.0, -5248423.0, 4021290.0),
|
|
new Cartesian3(-2539788.0, -4724797.0, 3620093.0),
|
|
],
|
|
outTangents: [
|
|
new Cartesian3(1125196, -161816, 270551),
|
|
new Cartesian3(-996690.5, -365906.5, 184028.5),
|
|
new Cartesian3(-2096917, 48379.5, -292683.5),
|
|
new Cartesian3(-890902.5, 408999.5, -447115),
|
|
],
|
|
inTangents: [
|
|
new Cartesian3(-1993381, -731813, 368057),
|
|
new Cartesian3(-4193834, 96759, -585367),
|
|
new Cartesian3(-1781805, 817999, -894230),
|
|
new Cartesian3(1165345, 112641, 47281),
|
|
],
|
|
});
|
|
|
|
var p0 = hs.points[0];
|
|
var p1 = hs.points[1];
|
|
var pT0 = hs.outTangents[0];
|
|
var pT1 = hs.inTangents[0];
|
|
var interpolate = createHermiteBasis(p0, pT0, p1, pT1);
|
|
|
|
var granularity = 0.1;
|
|
for (var i = 0.0; i < 1.0; i = i + granularity) {
|
|
expect(hs.evaluate(i)).toEqualEpsilon(
|
|
interpolate(i),
|
|
CesiumMath.EPSILON3
|
|
);
|
|
}
|
|
});
|
|
|
|
it("createC1 throws without points", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createC1();
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createC1 throws when control points length is less than 2", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createC1({
|
|
points: [Cartesian3.ZERO],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createC1 throws without times", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createC1({
|
|
points: points,
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createC1 throws when times.length is not equal to points.length", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createC1({
|
|
points: points,
|
|
times: [0.0, 1.0],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createC1 throws without tangents", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createC1({
|
|
points: points,
|
|
times: times,
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createC1 throws when tangents.length is not equal to times.length", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createC1({
|
|
points: points,
|
|
times: times,
|
|
tangents: [Cartesian3.ZERO],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("C1 spline", function () {
|
|
var times = [0.0, 1.0, 3.0, 4.5, 6.0];
|
|
var points = [
|
|
new Cartesian3(1235398.0, -4810983.0, 4146266.0),
|
|
new Cartesian3(1372574.0, -5345182.0, 4606657.0),
|
|
new Cartesian3(-757983.0, -5542796.0, 4514323.0),
|
|
new Cartesian3(-2821260.0, -5248423.0, 4021290.0),
|
|
new Cartesian3(-2539788.0, -4724797.0, 3620093.0),
|
|
];
|
|
|
|
var tangents = new Array(points.length);
|
|
tangents[0] = new Cartesian3(1125196, -161816, 270551);
|
|
for (var i = 1; i < tangents.length - 1; ++i) {
|
|
tangents[i] = Cartesian3.multiplyByScalar(
|
|
Cartesian3.subtract(points[i + 1], points[i - 1], new Cartesian3()),
|
|
0.5,
|
|
new Cartesian3()
|
|
);
|
|
}
|
|
tangents[tangents.length - 1] = new Cartesian3(1165345, 112641, 47281);
|
|
|
|
var interpolate = createHermiteBasis(
|
|
points[0],
|
|
tangents[0],
|
|
points[1],
|
|
tangents[1]
|
|
);
|
|
var hs = HermiteSpline.createC1({
|
|
times: times,
|
|
points: points,
|
|
tangents: tangents,
|
|
});
|
|
|
|
var granularity = 0.1;
|
|
for (var j = times[0]; j < times[1]; j = j + granularity) {
|
|
expect(hs.evaluate(j)).toEqualEpsilon(
|
|
interpolate(j),
|
|
CesiumMath.EPSILON3
|
|
);
|
|
}
|
|
});
|
|
|
|
it("createNaturalCubic throws without points", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createNaturalCubic();
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createNaturalCubic throws when control points length is less than 2", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createNaturalCubic({
|
|
points: [Cartesian3.ZERO],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createNaturalCubic throws without times", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createNaturalCubic({
|
|
points: points,
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createNaturalCubic throws when times.length is not equal to points.length", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createNaturalCubic({
|
|
points: points,
|
|
times: [0.0, 1.0],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("natural cubic spline", function () {
|
|
points = [
|
|
new Cartesian3(1.0, 0.0, 0.0),
|
|
new Cartesian3(0.0, 1.0, CesiumMath.PI_OVER_TWO),
|
|
new Cartesian3(-1.0, 0.0, Math.PI),
|
|
new Cartesian3(0.0, -1.0, CesiumMath.THREE_PI_OVER_TWO),
|
|
];
|
|
|
|
var p0Tangent = new Cartesian3(-0.87, 1.53, 1.57);
|
|
var p1Tangent = new Cartesian3(-1.27, -0.07, 1.57);
|
|
|
|
var interpolate = createHermiteBasis(
|
|
points[0],
|
|
p0Tangent,
|
|
points[1],
|
|
p1Tangent
|
|
);
|
|
var hs = HermiteSpline.createNaturalCubic({
|
|
points: points,
|
|
times: times,
|
|
});
|
|
|
|
var granularity = 0.1;
|
|
for (var i = times[0]; i < times[1]; i = i + granularity) {
|
|
expect(hs.evaluate(i)).toEqualEpsilon(
|
|
interpolate(i),
|
|
CesiumMath.EPSILON3
|
|
);
|
|
}
|
|
});
|
|
|
|
it("createClampedCubic throws without points", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createClampedCubic();
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createClampedCubic throws when control points length is less than 2", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createClampedCubic({
|
|
points: [Cartesian3.ZERO],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createClampedCubic throws without times", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createClampedCubic({
|
|
points: points,
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createClampedCubic throws when times.length is not equal to points.length", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createClampedCubic({
|
|
points: points,
|
|
times: [0.0, 1.0],
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createClampedCubic throws without firstTangent", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createClampedCubic({
|
|
points: points,
|
|
times: times,
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("createClampedCubic throws without lastTangent", function () {
|
|
expect(function () {
|
|
return HermiteSpline.createClampedCubic({
|
|
points: points,
|
|
times: times,
|
|
firstTangent: Cartesian3.ZERO,
|
|
});
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("clamped cubic spline", function () {
|
|
points = [
|
|
new Cartesian3(1.0, 0.0, 0.0),
|
|
new Cartesian3(0.0, 1.0, CesiumMath.PI_OVER_TWO),
|
|
new Cartesian3(-1.0, 0.0, Math.PI),
|
|
new Cartesian3(0.0, -1.0, CesiumMath.THREE_PI_OVER_TWO),
|
|
];
|
|
|
|
var firstTangent = new Cartesian3(0.0, 1.0, 0.0);
|
|
var lastTangent = new Cartesian3(1.0, 0.0, 0.0);
|
|
|
|
var p0Tangent = firstTangent;
|
|
var p1Tangent = new Cartesian3(-1.53, 0.13, 1.88);
|
|
|
|
var interpolate = createHermiteBasis(
|
|
points[0],
|
|
p0Tangent,
|
|
points[1],
|
|
p1Tangent
|
|
);
|
|
var hs = HermiteSpline.createClampedCubic({
|
|
points: points,
|
|
times: times,
|
|
firstTangent: firstTangent,
|
|
lastTangent: lastTangent,
|
|
});
|
|
|
|
var granularity = 0.1;
|
|
for (var i = points[0].time; i < points[1].time; i = i + granularity) {
|
|
expect(hs.evaluate(i)).toEqualEpsilon(
|
|
interpolate(i),
|
|
CesiumMath.EPSILON3
|
|
);
|
|
}
|
|
});
|
|
|
|
it("evaluate fails with undefined time", function () {
|
|
var hs = HermiteSpline.createNaturalCubic({
|
|
points: points,
|
|
times: times,
|
|
});
|
|
expect(function () {
|
|
hs.evaluate();
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("evaluate fails with time out of range", function () {
|
|
var hs = HermiteSpline.createNaturalCubic({
|
|
points: points,
|
|
times: times,
|
|
});
|
|
expect(function () {
|
|
hs.evaluate(times[0] - 1.0);
|
|
}).toThrowDeveloperError();
|
|
});
|
|
|
|
it("evaluate with result parameter", function () {
|
|
var hs = HermiteSpline.createNaturalCubic({
|
|
points: points,
|
|
times: times,
|
|
});
|
|
var result = new Cartesian3();
|
|
var point = hs.evaluate(times[0], result);
|
|
expect(point).toBe(result);
|
|
expect(result).toEqual(points[0]);
|
|
});
|
|
|
|
it("createNaturalCubic with 2 control points defaults to lerp", function () {
|
|
points = points.slice(0, 2);
|
|
times = times.slice(0, 2);
|
|
|
|
var hs = HermiteSpline.createNaturalCubic({
|
|
points: points,
|
|
times: times,
|
|
});
|
|
|
|
var t = (times[0] + times[1]) * 0.5;
|
|
expect(hs.evaluate(t)).toEqual(
|
|
Cartesian3.lerp(points[0], points[1], t, new Cartesian3())
|
|
);
|
|
});
|
|
|
|
it("createClampedCubic with 2 control points defaults to lerp", function () {
|
|
points = points.slice(0, 2);
|
|
times = times.slice(0, 2);
|
|
|
|
var hs = HermiteSpline.createNaturalCubic({
|
|
points: points,
|
|
times: times,
|
|
});
|
|
|
|
var t = (times[0] + times[1]) * 0.5;
|
|
expect(hs.evaluate(t)).toEqual(
|
|
Cartesian3.lerp(points[0], points[1], t, new Cartesian3())
|
|
);
|
|
});
|
|
});
|