com.graphbuilder.curve.BinaryCurveApproximationAlgorithm Maven / Gradle / Ivy
Show all versions of curvesapi Show documentation
/*
* Copyright (c) 2005, Graph Builder
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* * Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* * Neither the name of Graph Builder nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package com.graphbuilder.curve;
import com.graphbuilder.geom.Geom;
/**
The binary curve approximation algorithm is an algorithm designed to approximate a ParametricCurve using as few
points as possible but keeping the overall visual appearance of the curve smooth. The number of points used and the
visual appearance are not guaranteed, but the results are usually very good.
Users that do not plan on implementing their own ParametricCurves do not need to use this class.
There is one static method called genPts that takes a ParametricCurve, an interval [t_min, t_max], and a MultiPath
as parameters. The algorithm uses the ParametricCurve to generate points in the range [t_min, t_max] and append the
points in order to the MultiPath. Figure 1 shows the basic idea of how the algorithm works.
The algorithm works by evaluating the parametric midpoint (m) of two points and computing the distance (dist)
from the parametric midpoint to line segment formed by the two points. That distance is then compared to the
threshold distance, otherwise called the "flatness" of the curve. The algorithm continues to subdivide the curve
until dist < flatness is reached. However, that is not all there is to say. Figure 2 illustrates the problem of
this approach.
In some cases, the evaluated midpoint is on or near the line segment. To solve this, the idea of a sample limit
is used. The sample limit specifies how many additional subdivisions to perform once the condition dist < flatness
is reached. If any of the additional subdivisions does not meet the dist < flatness condition, then the algorithm
continues from the point that did not meet the condition. The closer the flatness value to 0 and the higher the
sample limit both increase the number of points appended to the multi-path.
The idea of a sample limit is adequate for curves that generate points on the curve using a fixed number of
control-points in sections. Curves that have this property have a fixed number of inflection points per section.
For example, the CubicBSpline generates itself in sections (considering 4 points at a time) but the BSpline with
degree 3 only has one section. For curves that do not have this property, the sample-limit can be specified.
Note: As the sample limit is increased, it becomes more difficult to produce a point arrangement that causes a
problem.
Note: The algorithm is designed so that the first point appended to the multi-path will be eval(t_min) and the
last point appended to the multi-path will be eval(t_max). If the result of computing the distance from the line
segment to the evaluated point is NaN or Infinity then the algorithm aborts by throwing a RuntimeException.
@see com.graphbuilder.curve.Curve
@see com.graphbuilder.curve.ParametricCurve
@see com.graphbuilder.curve.MultiPath
*/
public final class BinaryCurveApproximationAlgorithm {
private BinaryCurveApproximationAlgorithm() {}
/**
Appends a sequence of points to the multi-path using the lineTo method exclusively. The flatness and the
dimension of the curve are determined by the multi-path's flatness and dimension. The sample limit is
determined by the parametric curve's sample limit.
@throws IllegalArgumentException If t_min > t_max.
*/
public static void genPts(ParametricCurve pc, double t_min, double t_max, MultiPath mp) {
if (t_min > t_max)
throw new IllegalArgumentException("t_min <= t_max required.");
int n = mp.getDimension();
double t1 = t_min;
double t2 = t_max;
double[][] stack = new double[10][];
int count = 0;
double[] rdy = new double[n + 1];
rdy[n] = t1;
pc.eval(rdy);
double[] p = new double[n + 1];
p[n] = t2;
pc.eval(p);
stack[count++] = p;
double[][] limit = new double[pc.getSampleLimit()][];
double flatSq = mp.getFlatness() * mp.getFlatness();
double[] d = new double[n + 1]; // result array for the Geom.ptLineSegDistSq method
while (true) {
double m = (t1 + t2) / 2;
double[] pt = new double[n + 1];
pt[n] = m;
pc.eval(pt);
double dist = Geom.ptSegDistSq(rdy, stack[count-1], pt, d, n);
// an infinite loop will happen if the following is not checked
if (Double.isNaN(dist) || Double.isInfinite(dist)) {
String msg = "NaN or infinity resulted from calling the eval method of the " +
pc.getClass().getName() + " class.";
throw new RuntimeException(msg);
}
boolean flag = false;
if (dist < flatSq) {
int i = 0;
double mm = 0;
for (; i < limit.length; i++) {
mm = (t1 + m) / 2;
double[] q = new double[n + 1];
limit[i] = q;
q[n] = mm;
pc.eval(q);
if (Geom.ptSegDistSq(rdy, pt, q, d, n) >= flatSq)
break;
else
m = mm;
}
if (i == limit.length)
flag = true;
else {
stack = checkSpace(stack, count);
stack[count++] = pt;
for (int j = 0; j <= i; j++) {
stack = checkSpace(stack, count);
stack[count++] = limit[j];
}
t2 = mm;
}
}
if (flag) {
mp.lineTo(rdy);
mp.lineTo(pt);
rdy = stack[--count];
if (count == 0) break;
pt = stack[count - 1];
t1 = t2;
t2 = pt[n];
}
else if (t2 > m) { // case: dist >= flatSq
stack = checkSpace(stack, count);
stack[count++] = pt;
t2 = m;
}
}
mp.lineTo(rdy);
}
private static double[][] checkSpace(double[][] stack, int size) {
if (size == stack.length) {
double[][] arr = new double[2 * size][];
for (int i = 0; i < size; i++)
arr[i] = stack[i];
return arr;
}
return stack;
}
}