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

eu.mihosoft.vrl.v3d.Extrude Maven / Gradle / Ivy

There is a newer version: 2.0.7
Show newest version
/**
 * Extrude.java
 *
 * Copyright 2014-2014 Michael Hoffer [email protected]. All rights
 * reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * 1. Redistributions of source code must retain the above copyright notice,
 * this list of conditions and the following disclaimer.
 *
 * 2. 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.
 *
 * THIS SOFTWARE IS PROVIDED BY Michael Hoffer [email protected] "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 Michael Hoffer [email protected] 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.
 *
 * The views and conclusions contained in the software and documentation are
 * those of the authors and should not be interpreted as representing official
 * policies, either expressed or implied, of Michael Hoffer
 * [email protected].
 */
package eu.mihosoft.vrl.v3d;

import java.util.ArrayList;
import com.piro.bezier.BezierPath;
import eu.mihosoft.vrl.v3d.Transform;
import eu.mihosoft.vrl.v3d.svg.*;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;

//import javax.vecmath.Vector3d;

import eu.mihosoft.vrl.v3d.ext.org.poly2tri.PolygonUtil;
import eu.mihosoft.vrl.v3d.ext.quickhull3d.HullUtil;

// TODO: Auto-generated Javadoc
/**
 * Extrudes concave and convex polygons.
 *
 * @author Michael Hoffer <[email protected]>
 */
public class Extrude {
	private static IExtrusion extrusionEngine = new IExtrusion() {
		/**
		 * Extrudes the specified path (convex or concave polygon without holes or
		 * intersections, specified in CCW) into the specified direction.
		 *
		 * @param dir    direction
		 * @param points path (convex or concave polygon without holes or intersections)
		 *
		 * @return a CSG object that consists of the extruded polygon
		 */
		public CSG points(Vector3d dir, List points) {

			List newList = new ArrayList<>(points);

			return extrude(dir, Polygon.fromPoints(toCCW(newList)));
		}

		/**
		 * Extrude.
		 *
		 * @param dir      the dir
		 * @param polygon1 the polygon1
		 * @return the csg
		 */
		public CSG extrude(Vector3d dir, Polygon polygon1) {

			return monotoneExtrude(dir, polygon1);
		}

		private CSG monotoneExtrude(Vector3d dir, Polygon polygon1) {
			List newPolygons = new ArrayList<>();
			CSG extrude;
			//polygon1=polygon1.flipped();
			newPolygons.addAll(PolygonUtil.concaveToConvex(polygon1.flipped()));
			Polygon polygon2 = polygon1.translated(dir);

			int numvertices = polygon1.vertices.size();
			//com.neuronrobotics.sdk.common.Log.error("Building Polygon "+polygon1.getPoints().size());
			for (int i = 0; i < numvertices; i++) {

				int nexti = (i + 1) % numvertices;

				Vector3d bottomV1 = polygon1.vertices.get(i).pos;
				Vector3d topV1 = polygon2.vertices.get(i).pos;
				Vector3d bottomV2 = polygon1.vertices.get(nexti).pos;
				Vector3d topV2 = polygon2.vertices.get(nexti).pos;
				double distance = bottomV1.minus(bottomV2).magnitude();
				if(Math.abs(distance) pPoints = Arrays.asList(bottomV2, topV2, topV1, bottomV1);
				try {
					newPolygons.add(Polygon.fromPoints(pPoints, polygon1.getStorage()));
				}catch(Exception ex) {
					//com.neuronrobotics.sdk.common.Log.error("Polygon has problems: ");
					ex.printStackTrace();
				}
			}

			polygon2 = polygon2.flipped();
			List topPolygons = PolygonUtil.concaveToConvex(polygon2.flipped());

			newPolygons.addAll(topPolygons);
			extrude = CSG.fromPolygons(newPolygons);

			return extrude;
		}

		@Override
		public CSG extrude(Vector3d dir, List points) {
			return points(dir, points);
		}
	};

	/**
	 * Instantiates a new extrude.
	 */
	private Extrude() {
		throw new AssertionError("Don't instantiate me!", null);
	}

	public static CSG polygons(Polygon polygon1, Number zDistance) {
		return polygons(polygon1, polygon1.transformed(new Transform().movez(zDistance)));
	}

	public static CSG polygons(Polygon polygon1, Polygon polygon2) {
		// if(!isCCW(polygon1)) {
		// polygon1=Polygon.fromPoints(toCCW(polygon1.getPoints()));
		// }
		// if(!isCCW(polygon2)) {
		// polygon2=Polygon.fromPoints(toCCW(polygon2.getPoints()));
		// }

		List newPolygons = new ArrayList<>();
		CSG extrude;
		newPolygons.addAll(PolygonUtil.concaveToConvex(polygon1.flipped()));
		if (polygon1.vertices.size() != polygon2.vertices.size()) {
			throw new RuntimeException("These polygons do not match");
		}

		int numvertices = polygon1.vertices.size();
		for (int i = 0; i < numvertices; i++) {

			int nexti = (i + 1) % numvertices;

			Vector3d bottomV1 = polygon1.vertices.get(i).pos;
			Vector3d topV1 = polygon2.vertices.get(i).pos;
			Vector3d bottomV2 = polygon1.vertices.get(nexti).pos;
			Vector3d topV2 = polygon2.vertices.get(nexti).pos;

			List pPoints = Arrays.asList(bottomV2, topV2, topV1, bottomV1);

			newPolygons.add(Polygon.fromPoints(pPoints, polygon1.getStorage()));

		}

		polygon2 = polygon2.flipped();
		List topPolygons = PolygonUtil.concaveToConvex(polygon2.flipped());

		newPolygons.addAll(topPolygons);
		extrude = CSG.fromPolygons(newPolygons);

		return extrude;
	}

	public static ArrayList polygons(eu.mihosoft.vrl.v3d.Polygon polygon1, ArrayList transforms) {
		if (transforms.size() == 1)
			transforms.add(0, new Transform());
		polygon1 = Polygon.fromPoints(toCCW(polygon1.getPoints()));
		if (transforms.size() < 2) {
			transforms.add(0, new Transform());
		}
		ArrayList parts = new ArrayList<>();
		Transform transform = new Transform();
		// transform.rotY(90);
		for (int i = 0; i < transforms.size() - 1; i++) {
			CSG tmp = polygons(polygon1.transformed(transform).transformed(transforms.get(i)),
					polygon1.transformed(transform).transformed(transforms.get(i + 1)));
			parts.add(tmp);
		}
		return parts;

	}

	public static ArrayList polygons(eu.mihosoft.vrl.v3d.Polygon polygon1, Transform... transformparts) {

		return polygons(polygon1, (ArrayList) Arrays.asList(transformparts));

	}

	public static CSG points(Vector3d dir, List points) {

		return getExtrusionEngine().extrude(dir, points);
	}

	/**
	 * Extrudes the specified path (convex or concave polygon without holes or
	 * intersections, specified in CCW) into the specified direction.
	 *
	 * @param dir    direction
	 * @param points path (convex or concave polygon without holes or intersections)
	 *
	 * @return a CSG object that consists of the extruded polygon
	 */
	public static CSG points(Vector3d dir, Vector3d... points) {

		return points(dir, Arrays.asList(points));
	}

	/**
	 * To ccw.
	 *
	 * @param points the points
	 * @return the list
	 */
	public static List toCCW(List points) {

		List result = new ArrayList<>(points);

		if (!isCCW(Polygon.fromPoints(result))) {
			Collections.reverse(result);
		}

		return result;
	}

	/**
	 * To cw.
	 *
	 * @param points the points
	 * @return the list
	 */
	static List toCW(List points) {

		List result = new ArrayList<>(points);

		if (isCCW(Polygon.fromPoints(result))) {
			Collections.reverse(result);
		}

		return result;
	}

	/**
	 * Checks if is ccw.
	 *
	 * @param polygon the polygon
	 * @return true, if is ccw
	 */
	public static boolean isCCW(Polygon polygon) {

		// thanks to Sepp Reiter for explaining me the algorithm!

		if (polygon.vertices.size() < 3) {
			throw new IllegalArgumentException("Only polygons with at least 3 vertices are supported!");
		}

		// search highest left vertex
		int highestLeftVertexIndex = 0;
		Vertex highestLeftVertex = polygon.vertices.get(0);
		for (int i = 0; i < polygon.vertices.size(); i++) {
			Vertex v = polygon.vertices.get(i);

			if (v.pos.y > highestLeftVertex.pos.y) {
				highestLeftVertex = v;
				highestLeftVertexIndex = i;
			} else if (v.pos.y == highestLeftVertex.pos.y && v.pos.x < highestLeftVertex.pos.x) {
				highestLeftVertex = v;
				highestLeftVertexIndex = i;
			}
		}

		// determine next and previous vertex indices
		int nextVertexIndex = (highestLeftVertexIndex + 1) % polygon.vertices.size();
		int prevVertexIndex = highestLeftVertexIndex - 1;
		if (prevVertexIndex < 0) {
			prevVertexIndex = polygon.vertices.size() - 1;
		}
		Vertex nextVertex = polygon.vertices.get(nextVertexIndex);
		Vertex prevVertex = polygon.vertices.get(prevVertexIndex);

		// edge 1
		double a1 = normalizedX(highestLeftVertex.pos, nextVertex.pos);

		// edge 2
		double a2 = normalizedX(highestLeftVertex.pos, prevVertex.pos);

		// select vertex with lowest x value
		int selectedVIndex;

		if (a2 > a1) {
			selectedVIndex = nextVertexIndex;
		} else {
			selectedVIndex = prevVertexIndex;
		}

		if (selectedVIndex == 0 && highestLeftVertexIndex == polygon.vertices.size() - 1) {
			selectedVIndex = polygon.vertices.size();
		}

		if (highestLeftVertexIndex == 0 && selectedVIndex == polygon.vertices.size() - 1) {
			highestLeftVertexIndex = polygon.vertices.size();
		}

		// indicates whether edge points from highestLeftVertexIndex towards
		// the sel index (ccw)
		return selectedVIndex > highestLeftVertexIndex;

	}

	/**
	 * Normalized x.
	 *
	 * @param v1 the v1
	 * @param v2 the v2
	 * @return the double
	 */
	private static double normalizedX(Vector3d v1, Vector3d v2) {
		Vector3d v2MinusV1 = v2.minus(v1);

		return v2MinusV1.dividedBy(v2MinusV1.magnitude()).times(Vector3d.X_ONE).x;
	}

	public static IExtrusion getExtrusionEngine() {
		return extrusionEngine;
	}

	public static void setExtrusionEngine(IExtrusion extrusionEngine) {
		Extrude.extrusionEngine = extrusionEngine;
	}

	public static CSG byPath(List> points, double height) {

		return byPath(points, height, 200);

	}

	public static CSG byPath(List> points, double height, int resolution) {
		ArrayList trPath = pathToTransforms(points, resolution);
		ArrayList finalPath = new ArrayList<>();
		for (Transform tr : trPath) {
			javax.vecmath.Vector3d t1 = new javax.vecmath.Vector3d();
			tr.getInternalMatrix().get(t1);
			Vector3d finalPoint = new Vector3d(t1.x, t1.y, 0);
			finalPath.add(finalPoint);
		}
		// showEdges(finalPath,(double)0.0,javafx.scene.paint.Color.RED)
		// println "Path size = " +finalPath.size()
		// List p = Polygon.fromConcavePoints(finalPath)
		// for(Polygon pl:p)
		// BowlerStudioController.getBowlerStudio()addObject(pl,null)
		// return new Cube(height).toCSG()
		return Extrude.points(new Vector3d(0, 0, height), finalPath);
	}

	public static ArrayList pathToTransforms(List> points, int resolution) {

		Vector3d start = points.get(0).get(0);
		String pathStringA = "M " + start.x + "," + start.y;
		String pathStringB = pathStringA;

		for (List sections : points) {
			if (sections.size() == 4) {
				Vector3d controlA = sections.get(1);
				Vector3d controlB = sections.get(2);
				Vector3d endPoint = sections.get(3);
				/*
				 * ArrayList controlA = (ArrayList)
				 * Arrays.asList(sections.get(1).x - start.get(0), sections.get(1).y -
				 * start.get(1), sections.get(1).z - start.get(2));
				 * 
				 * ArrayList controlB = (ArrayList)
				 * Arrays.asList(sections.get(2).x - start.get(0), sections.get(2).y -
				 * start.get(1), sections.get(2).z - start.get(2)); ; ArrayList endPoint
				 * = (ArrayList) Arrays.asList(sections.get(3).x - start.get(0),
				 * sections.get(3).y - start.get(1), sections.get(3).z - start.get(2)); ;
				 */

				pathStringA += ("C " + controlA.x + "," + controlA.y + " " + controlB.x + "," + controlB.y + " "
						+ endPoint.x + "," + endPoint.y + "\n");
				pathStringB += ("C " + controlA.x + "," + controlA.z + " " + controlB.x + "," + controlB.z + " "
						+ endPoint.x + "," + endPoint.z + "\n");
				// start.set(0, sections.get(3).x);
				// start.set(1, sections.get(3).y);
				// start.set(2,sections.get(3).z);

			} else if (sections.size() == 1) {

				pathStringA += "L " + (double) sections.get(0).x + "," + (double) sections.get(0).y + "\n";
				pathStringB += "L " + (double) sections.get(0).x + "," + (double) sections.get(0).z + "\n";
				// start.set(0, sections.get(0).x);
				// start.set(1, sections.get(0).y);
				// start.set(2, sections.get(0).z);
			}
		}
		// println "A string = " +pathStringA
		// println "B String = " +pathStringB
		BezierPath path = new BezierPath();
		path.parsePathString(pathStringA);
		BezierPath path2 = new BezierPath();
		path2.parsePathString(pathStringB);

		return bezierToTransforms(path, path2, resolution, null, null);
	}

	public static ArrayList moveAlongProfile(CSG object, List> points, int resolution) {

		return Extrude.move(object, pathToTransforms(points, resolution));
	}

	public static ArrayList bezierToTransforms(Vector3d controlA, Vector3d controlB, Vector3d endPoint,
			int iterations) {
		BezierPath path = new BezierPath();
		path.parsePathString("C " + controlA.x + "," + controlA.y + " " + controlB.x + "," + controlB.y + " "
				+ endPoint.x + "," + endPoint.y);
		BezierPath path2 = new BezierPath();
		path2.parsePathString("C " + controlA.x + "," + controlA.z + " " + controlB.x + "," + controlB.z + " "
				+ endPoint.x + "," + endPoint.z);

		return bezierToTransforms(path, path2, iterations, controlA, controlB);
	}

	public static ArrayList bezierToTransforms(List parts, int iterations) {
		// //com.neuronrobotics.sdk.common.Log.error("Bezier type "+parts.size());
		if (parts.size() == 3)
			return bezierToTransforms(parts.get(0), parts.get(1), parts.get(2), iterations);
		if (parts.size() == 2)
			return bezierToTransforms(parts.get(0), parts.get(0), parts.get(1), parts.get(1), iterations);
		if (parts.size() == 1)
			return bezierToTransforms(new Vector3d(0, 0, 0), new Vector3d(0, 0, 0), parts.get(0), parts.get(0),
					iterations);
		return bezierToTransforms(parts.get(0), parts.get(1), parts.get(2), parts.get(3), iterations);
	}

	private Transform toTransform() {

		return null;
	}

	public static ArrayList bezierToTransforms(BezierPath pathA, BezierPath pathB, int iterations,
			Vector3d controlA, Vector3d controlB) {
		double d = 1.0 / (double) iterations;
		ArrayList p = new ArrayList();
		Vector3d pointAStart = pathA.eval(0);
		Vector3d pointBStart = pathB.eval(0);
		double x = pointAStart.x, y = pointAStart.y, z = pointBStart.y;
		double lastx = x, lasty = y, lastz = z;
		// float min = (float) 0.0001;
		int startIndex = 0;
		if (controlA != null) {
			startIndex = 1;
			double ydiff = controlA.y - y;
			double zdiff = controlA.z - z;
			double xdiff = controlA.x - x;
			double rise = zdiff;
			double run = Math.sqrt((ydiff * ydiff) + (xdiff * xdiff));
			double rotz = 90 - Math.toDegrees(Math.atan2(xdiff, ydiff));
			// //com.neuronrobotics.sdk.common.Log.error("Rot z = "+rotz+" x="+xdiff+" y="+ydiff);
			double roty = Math.toDegrees(Math.atan2(rise, run));
			Transform t = new Transform();
			t.translateX(x);
			t.translateY(y);
			t.translateZ(z);
			t.rotZ(-rotz);
			t.rotY(roty);
			p.add(t);
		}
		Transform t;
		double ydiff;
		double zdiff;
		double xdiff;
		double rise;
		double run;
		double rotz;
		double roty;
		for (int i = startIndex; i < iterations - 1; i++) {
			float pathFunction = (float) (((float) i) / ((float) (iterations - 1)));

			Vector3d pointA = pathA.eval(pathFunction);
			Vector3d pointB = pathB.eval(pathFunction);

			x = pointA.x;
			y = pointA.y;
			z = pointB.y;

			t = new Transform();
			t.translateX(x);
			t.translateY(y);
			t.translateZ(z);

			Vector3d pointAEst = pathA.eval((float) (pathFunction + d));
			Vector3d pointBEst = pathB.eval((float) (pathFunction + d));
			double xest = pointAEst.x;
			double yest = pointAEst.y;
			double zest = pointBEst.y;
			ydiff = yest - y;
			zdiff = zest - z;
			xdiff = xest - x;
			// t.rotX(45-Math.toDegrees(Math.atan2(zdiff,ydiff)))

			rise = zdiff;
			run = Math.sqrt((ydiff * ydiff) + (xdiff * xdiff));
			rotz = 90 - Math.toDegrees(Math.atan2(xdiff, ydiff));
			// //com.neuronrobotics.sdk.common.Log.error("Rot z = "+rotz+" x="+xdiff+" y="+ydiff);
			roty = Math.toDegrees(Math.atan2(rise, run));

			t.rotZ(-rotz);
			t.rotY(roty);
			// if(i==0)
			// //com.neuronrobotics.sdk.common.Log.error( " Tr = "+x+" "+y+" "+z+" path = "+pathFunction);
			// println "z = "+rotz+" y = "+roty
			p.add(t);
			lastx = x;
			lasty = y;
			lastz = z;
		}

		Vector3d pointA = pathA.eval((float) 1);
		Vector3d pointB = pathB.eval((float) 1);

		x = pointA.x;
		y = pointA.y;
		z = pointB.y;
		t = new Transform();
		t.translateX(x);
		t.translateY(y);
		t.translateZ(z);
		if (controlB != null) {
			lastx = controlB.x;
			lasty = controlB.y;
			lastz = controlB.z;
		}

		ydiff = y - lasty;
		zdiff = z - lastz;
		xdiff = x - lastx;

		rise = zdiff;
		run = Math.sqrt((ydiff * ydiff) + (xdiff * xdiff));
		rotz = 90 - Math.toDegrees(Math.atan2(xdiff, ydiff));
		roty = Math.toDegrees(Math.atan2(rise, run));

		t.rotZ(-rotz);
		t.rotY(roty);
		p.add(t);

		return p;
	}

	public static ArrayList bezierToTransforms(Vector3d start, Vector3d controlA, Vector3d controlB,
			Vector3d endPoint, int iterations) {
		String startString = "M " + start.x + "," + start.y + "\n" + "C " + controlA.x + "," + controlA.y + " "
				+ controlB.x + "," + controlB.y + " " + endPoint.x + "," + endPoint.y;
		String b = "M " + start.x + "," + start.z + "\n" + "C " + controlA.x + "," + controlA.z + " " + controlB.x + ","
				+ controlB.z + " " + endPoint.x + "," + endPoint.z;
		// println "Start = "+startString
		BezierPath path = new BezierPath();
		path.parsePathString(startString);
		BezierPath path2 = new BezierPath();
		path2.parsePathString(b);
		// newParts.remove(parts.size()-1)
		// newParts.remove(0)
		// //com.neuronrobotics.sdk.common.Log.error("Parsing "+startString+" \nand\n"+b);
		return bezierToTransforms(path, path2, iterations, controlA, controlB);
	}

	public static ArrayList revolve(CSG slice, double radius, int numSlices) {
		return revolve(slice, radius, 360.0, null, numSlices);
	}

	public static ArrayList revolve(Polygon poly, int numSlices) {
		return revolve(poly, 0, numSlices);
	}

	public static ArrayList revolve(Polygon poly, double radius, int numSlices) {
		ArrayList parts = new ArrayList();
		ArrayList slices = new ArrayList();

		for (int i = 0; i < numSlices; i++) {
			double angle = 360.0 / ((double) numSlices) * ((double) i);
			slices.add(poly.transformed(new Transform().movex(radius).roty(angle)));
		}
		for (int i = 0; i < slices.size(); i++) {
			int next = i + 1;
			if (next == slices.size())
				next = 0;
			// println "Extruding "+i+" to "+next
			parts.add(Extrude.polygons(slices.get(i), slices.get(next)));
		}

		return parts;
	}

	public static ArrayList revolve(CSG slice, double radius, double archLen, int numSlices) {
		return revolve(slice, radius, archLen, null, numSlices);
	}

	public static ArrayList revolve(CSG slice, double radius, double archLen, List> points,
			int numSlices) {
		ArrayList parts = new ArrayList();
		double slices = (double) numSlices;
		double increment = archLen / slices;
		CSG slicePRofile = slice.movey(radius);

		for (double i = 0; i < (archLen + increment); i += increment) {
			parts.add(slicePRofile.rotz(i));
		}
		if (points != null) {
			ArrayList pathtransforms = pathToTransforms(points, numSlices);
			for (int i = 0; i < parts.size(); i++) {
				CSG sweep = parts.get(i).transformed(pathtransforms.get(i));
				parts.set(i, sweep);
			}
		}
		for (int i = 0; i < parts.size() - 1; i++) {

			CSG sweep = CSG.hullAll(parts.get(i), parts.get(i + 1));
			parts.set(i, sweep);
		}

		return parts;
	}

	public static ArrayList bezier(CSG slice, ArrayList controlA, ArrayList controlB,
			ArrayList endPoint, int numSlices) {
		ArrayList parts = new ArrayList();

		for (int i = 0; i < numSlices; i++) {
			parts.add(0, slice.clone());
		}
		return bezier(parts, controlA, controlB, endPoint);
	}

	public static ArrayList bezier(ArrayList s, ArrayList controlA, ArrayList controlB,
			ArrayList endPoint) {
		ArrayList slice = moveBezier(s, controlA, controlB, endPoint);

		for (int i = 0; i < slice.size() - 1; i++) {
			// Polygon p1 =Slice.slice(slice.get(i), new Transform(), 0).get(0);
			// Polygon p2 =Slice.slice(slice.get(i+1), new Transform(), 0).get(0);
			// CSG sweep = polygons(p1, p2);
			CSG sweep = HullUtil.hull(slice.get(i), slice.get(i + 1));

			slice.set(i, sweep);
		}
		return slice;
	}

	public static ArrayList hull(ArrayList s, ArrayList p) {
		ArrayList slice = move(s, p);
		for (int i = 0; i < slice.size() - 1; i++) {
			// Polygon p1 =Slice.slice(slice.get(i), new Transform(), 0).get(0);
			// Polygon p2 =Slice.slice(slice.get(i+1), new Transform(), 0).get(0);
			// CSG sweep = polygons(p1, p2);
			CSG sweep = HullUtil.hull(slice.get(i), slice.get(i + 1));

			slice.set(i, sweep);
		}
		return slice;
	}

	public static ArrayList hull(CSG c, ArrayList p) {
		ArrayList s = new ArrayList<>();
		for (int i = 0; i < p.size(); i++) {
			s.add(c.clone());
		}
		return hull(s, p);
	}

	public static ArrayList linear(ArrayList s, ArrayList endPoint) {
		ArrayList start = (ArrayList) Arrays.asList(0.0, 0.0, 0.0);
		return bezier(s, start, endPoint, endPoint);
	}

	public static ArrayList linear(CSG s, ArrayList endPoint, int numSlices) {
		ArrayList start = (ArrayList) Arrays.asList(0.0, 0.0, 0.0);

		return bezier(s, start, endPoint, endPoint, numSlices);
	}

	public static ArrayList move(ArrayList slice, ArrayList p) {

		return CSG.move(slice, p);
	}

	public static ArrayList move(CSG slice, ArrayList p) {
		return slice.move(p);
	}

	public static ArrayList moveBezier(CSG slice, ArrayList controlA, ArrayList controlB,
			ArrayList endPoint, int numSlices) {
		ArrayList p = bezierToTransforms(fromDouble(controlA), fromDouble(controlB), fromDouble(endPoint),
				numSlices);

		return move(slice, p);
	}

	public static ArrayList moveBezier(ArrayList slice, ArrayList controlA,
			ArrayList controlB, ArrayList endPoint) {

		int numSlices = slice.size();
		ArrayList p = bezierToTransforms(fromDouble(controlA), fromDouble(controlB), fromDouble(endPoint),
				numSlices);
		return move(slice, p);

	}

	private static Vector3d fromDouble(ArrayList controlA) {
		return new Vector3d(controlA.get(0), controlA.get(1), controlA.get(2));
	}

	public static ArrayList moveBezier(CSG slice, BezierPath pathA, int numSlices) {
		Vector3d pointA = pathA.eval((float) 1.0);
		String zpath = "C 0,0 " + pointA.x + "," + pointA.y + " " + pointA.x + "," + pointA.y;
		BezierPath pathB = new BezierPath();
		pathB.parsePathString(zpath);

		return moveBezier(slice, pathA, pathB, numSlices);

	}

	public static ArrayList moveBezier(CSG slice, BezierPath pathA, BezierPath pathB, int iterations) {

		ArrayList p = bezierToTransforms(pathA, pathB, iterations, null, null);
		return move(slice, p);

	}

	public static Polygon toCCW(Polygon concave) {
		if (!isCCW(concave)) {
			List points = concave.getPoints();
			List result = new ArrayList<>(points);
			Collections.reverse(result);
			return Polygon.fromPoints(result);
		}
		return concave;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy