All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.graphbuilder.curve.MultiPath Maven / Gradle / Ivy

Go to download

The Apache Commons Codec package contains simple encoder and decoders for various formats such as Base64 and Hexadecimal. In addition to these widely used encoders and decoders, the codec package also maintains a collection of phonetic encoding utilities.

There is a newer version: 62
Show newest version
/*
* 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;

/**
A multi-path is a series of paths (sequence of connected points) in n-dimensions.  The points and
move types (MOVE_TO or LINE_TO) are stored using arrays.

@see com.graphbuilder.curve.Curve
@see com.graphbuilder.curve.ShapeMultiPath
*/
public class MultiPath {

	public static final Object MOVE_TO = new Object();
	public static final Object LINE_TO = new Object();

	private double[][] point = new double[2][0];
	private Object[] type = new Object[point.length];
	private int size = 0;

	private double flatness = 1.0;
	private final int dimension;

	/**
	Constructs a multi-path specifying the minimum required dimension of each point appended
	to this multi-path.

	@throws IllegalArgumentException If dimension <= 0.
	*/
	public MultiPath(int dimension) {
		if (dimension <= 0)
			throw new IllegalArgumentException("dimension > 0 required");

		this.dimension = dimension;
	}

	/**
	Returns the dimension. The dimension is used by the BinaryCurveApproximationAlgorithm
	to know what dimension of points to create.  If the dimension of the multi-path is greater
	than the dimension of the control points for a curve, then an ArrayIndexOutOfBoundsException
	will occur when the curve is appended to the MultiPath.
	*/
	public int getDimension() {
		return dimension;
	}

	/**
	Returns the flatness.  The flatness is used by the BinaryCurveApproximationAlgorithm to
	determine how closely the line segements formed by the points of this multi-path should
	approximate a given curve.  The default flatness value is 1.0.  When using curves in a
	graphics environment, the flatness usually inversely proportional to the scale.

	@see #setFlatness(double)
	*/
	public double getFlatness() {
		return flatness;
	}

	/**
	Sets the flatness.  As the flatness value gets closer to zero, the BinaryCurveApproximationAlgorithm
	generates more points.

	@throws IllegalArgumentException If the flatness is <= 0.
	@see #getFlatness()
	*/
	public void setFlatness(double f) {
		if (f <= 0)
			throw new IllegalArgumentException("flatness > 0 required");

		flatness = f;
	}

	/**
	Returns a reference to the point at the specified index.

	@see #set(int, double[])
	*/
	public double[] get(int index) {
		return point[index];
	}

	/**
	Sets the point at the specified index.

	@throws IllegalArgumentException If the point is null or the dimension of the point does not
	meet the dimension requirement specified in the constructor.
	@see #get(int)
	*/
	public void set(int index, double[] p) {
		if (p == null)
			throw new IllegalArgumentException("Point cannot be null.");

		if (p.length < dimension)
			throw new IllegalArgumentException("p.length >= dimension required");

		if (point[index] == null)
			throw new ArrayIndexOutOfBoundsException(index);

		point[index] = p;
	}

	/**
	Returns the type of the point at the specified index.  The type can other be MultiPath.MOVE_TO
	or MultiPath.LINE_TO.

	@see #setType(int, Object)
	*/
	public Object getType(int index) {
		if (type[index] == null)
			throw new ArrayIndexOutOfBoundsException(index);

		return type[index];
	}

	/**
	Sets the type of the point at the specified index.  The first point must always be MOVE_TO.

	@throws IllegalArgumentException If the specified type unknown or the specified index is 0 and type
	is not MOVE_TO.
	@see #getType(int)
	*/
	public void setType(int index, Object type) {
		if (type != MOVE_TO && type != LINE_TO)
			throw new IllegalArgumentException("unknown type");

		if (this.type[index] == null)
			throw new ArrayIndexOutOfBoundsException(index);

		if (index == 0 && type != MOVE_TO)
			throw new IllegalArgumentException("type[0] must always be MOVE_TO");

		this.type[index] = type;
	}

	/**
	Returns the size counter.

	@see #setNumPoints(int)
	*/
	public int getNumPoints() {
		return size;
	}

	/**
	Sets the size counter.  The size counter can be changed to any value as long as all points from index location
	0 (inclusive) to n (exclusive) are non-null.  Thus, the maximum value the size counter can be is the first
	null index location.  Changing the size counter does not remove any of the points.

	@see #getNumPoints()
	*/
	public void setNumPoints(int n) {
		if (n != 0 && point[n-1] == null)
			throw new ArrayIndexOutOfBoundsException(n);
		size = n;
	}

	/**
	Returns the capacity of the internal point array.
	*/
	public int getCapacity() {
		return point.length;
	}

	/**
	Checks that the point array has the specified capacity, otherwise the capacity of the point
	array is increased to be the maximum between twice the current capacity and the specified capacity.
	*/
	public void ensureCapacity(int capacity) {
		if (point.length < capacity) {
			int x = 2 * point.length;

			if (x < capacity)
				x = capacity;

			double[][] p2 = new double[x][];
			for (int i = 0; i < size; i++)
				p2[i] = point[i];

			Object[] t2 = new Object[x];
			for (int i = 0; i < size; i++)
				t2[i] = type[i];

			point = p2;
			type = t2;
		}
	}

	/**
	Creates a new point array of exact size, copying the points from the old array into the
	new one.
	*/
	public void trimArray() {
		if (size < point.length) {
			double[][] p2 = new double[size][];
			for (int i = 0; i < size; i++)
				p2[i] = point[i];

			Object[] t2 = new Object[size];
			for (int i = 0; i < size; i++)
				t2[i] = type[i];

			point = p2;
			type = t2;
		}
	}

	/**
	Appends a point of type LINE_TO.  If the size counter is 0 then the request is interpretted as a
	MOVE_TO request.

	@throws IllegalArgumentException If the point is null or the dimension of the point does not meet
	the dimension requirement specified in the constructor.
	@see #moveTo(double[])
	*/
	public void lineTo(double[] p) {
		append(p, LINE_TO);
	}

	/**
	Appends a point of type MOVE_TO.

	@throws IllegalArgumentException If the point is null or the dimension of the point does not meet the
	dimension requirement specified in the constructor.
	@see #lineTo(double[])
	*/
	public void moveTo(double[] p) {
		append(p, MOVE_TO);
	}

	private void append(double[] p, Object t) {
		if (p == null)
			throw new IllegalArgumentException("Point cannot be null.");

		if (p.length < dimension)
			throw new IllegalArgumentException("p.length >= dimension required");

		if (size == 0) t = MOVE_TO;

		ensureCapacity(size + 1);
		point[size] = p;
		type[size] = t;
		size++;
	}

	/**
	Computes the minimum distance^2 from the specified point to the line segments formed by the
	points of this multi-path.  If the size counter is 0 then the value returned is Double.MAX_VALUE.

	@throws IllegalArgumentException If the point is null or the length of the point is less than the
	dimension specified in the constructor.
	*/
	public double getDistSq(double[] p) {
		if (p == null)
			throw new IllegalArgumentException("Point cannot be null.");

		if (p.length < dimension)
			throw new IllegalArgumentException("p.length >= dimension required");

		int n = getNumPoints();

		if (n == 0)
			return Double.MAX_VALUE;

		double dist = Double.MAX_VALUE;

		double[] b = get(0);

		double[] c = new double[dimension + 1];

		for (int i = 1; i < n; i++) {
			double[] a = get(i);

			if (getType(i) == MultiPath.LINE_TO) {
				double d = Geom.ptSegDistSq(a, b, p, c, dimension);

				if (d < dist)
					dist = d;
			}
		}

		return dist;
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy