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

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

package eu.mihosoft.vrl.v3d;

import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;

import javafx.scene.shape.ClosePath;
import javafx.scene.shape.CubicCurveTo;
import javafx.scene.shape.LineTo;
import javafx.scene.shape.MoveTo;
import javafx.scene.shape.Path;
import javafx.scene.shape.PathElement;
import javafx.scene.shape.QuadCurveTo;
import javafx.scene.shape.Rectangle;
import javafx.scene.shape.Shape;
import javafx.scene.text.Font;
import javafx.scene.text.Text;
import java.util.stream.Collectors;
import java.util.stream.IntStream;
import javafx.beans.property.DoubleProperty;
import javafx.beans.property.SimpleDoubleProperty;
import javafx.scene.paint.Color;
import javafx.scene.shape.Path;
import javafx.scene.shape.PathElement;
import javafx.scene.shape.MoveTo;
import javafx.scene.shape.LineTo;
import javafx.scene.shape.CubicCurveTo;
import javafx.scene.shape.QuadCurveTo;
import javafx.scene.shape.ClosePath;
import javafx.scene.text.Text;
import eu.mihosoft.vrl.v3d.Vector3d;
import java.util.*;

// TODO: Auto-generated Javadoc
/**
 * The Class Text.
 */

@SuppressWarnings("restriction")
public class TextExtrude {
	private static final String default_font = "FreeSerif";
	private final static int POINTS_CURVE = 10;

	private final String text;
	// private List points;
	// HashSet unique = new HashSet();
	private Vector3d p0;
	private final List polis = new ArrayList<>();
	ArrayList sections = new ArrayList();
	ArrayList holes = new ArrayList();
	private double dir;

	class LineSegment {

		/*
		 * Given one single character in terms of Path, LineSegment stores a list of
		 * points that define the exterior of one of its polygons (!isHole). It can
		 * contain reference to one or several holes inside this polygon. Or it can
		 * define the perimeter of a hole (isHole), with no more holes inside.
		 */

		private boolean hole;
		private List points;
		private Path path;
		private Vector3d origen;
		private List holes = new ArrayList<>();
		private String letter;

		public LineSegment(String text) {
			letter = text;
		}

		public String getLetter() {
			return letter;
		}

		public void setLetter(String letter) {
			this.letter = letter;
		}

		public boolean isHole() {
			return hole;
		}

		public void setHole(boolean isHole) {
			this.hole = isHole;
		}

		public List getPoints() {
			return points;
		}

		public void setPoints(List points) {
			this.points = points;
		}

		public Path getPath() {
			return path;
		}

		public void setPath(Path path) {
			this.path = path;
		}

		public Vector3d getOrigen() {
			return origen;
		}

		public void setOrigen(Vector3d origen) {
			this.origen = origen;
		}

		public List getHoles() {
			return holes;
		}

		public void setHoles(List holes) {
			this.holes = holes;
		}

		public void addHole(LineSegment hole) {
			holes.add(hole);
		}

		@Override
		public String toString() {
			return "Poly{" + "points=" + points + ", path=" + path + ", origen=" + origen + ", holes=" + holes + '}';
		}
	}

	private TextExtrude(String text, Font font, double dir) {
		if (dir <= 0)
			throw new NumberFormatException("length can not be negative");
		this.dir = dir;
		// points = new ArrayList<>();
		this.text = text;
		Text textNode = new Text(text);
		textNode.setFont(font);

		// Convert Text to Path
		Path subtract = (Path) (Shape.subtract(textNode, new Rectangle(0, 0)));
		List> outlines = extractOutlines(subtract);
		double zOff = 0;
		for (List points : outlines) {
			boolean hole = Extrude.isCCW(Polygon.fromPoints(points));
			CSG newLetter = Extrude.points(new Vector3d(0, 0, dir), points).movez(zOff);

			if (!hole)
				sections.add(newLetter);
			else
				holes.add(newLetter);
			// zOff+=dir;

		}
//		// Convert Path elements into lists of points defining the perimeter
//		// (exterior or interior)
//		subtract.getElements().forEach(this::getPoints);

		for (int i = 0; i < sections.size(); i++) {
			for (CSG h : holes) {
				try {
					if (sections.get(i).touching(h)) {
						// println "Hole found "
						CSG nl = sections.get(i).difference(h);

						sections.set(i, nl);
					}
				} catch (Exception e) {

				}
			}
		}
	}

	/**
	 * Extrudes the specified path (convex or concave polygon without holes or
	 * intersections, specified in CCW) into the specified direction.
	 *
	 * @param dir  direction of extrusion
	 * @param text text
	 * @param font font configuration of the text
	 *
	 * @return a CSG object that consists of the extruded polygon
	 */
	@SuppressWarnings("restriction")
	public static ArrayList text(double dir, String text, Font font) {

		TextExtrude te = new TextExtrude(text, font, dir);

		return te.sections;
	}

	public List getLineSegment() {
		return polis;
	}

	public List getOffset() {
		return polis.stream().sorted((p1, p2) -> (int) (p1.getOrigen().x - p2.getOrigen().x))
				.map(LineSegment::getOrigen).collect(Collectors.toList());
	}

// Below is AI slop
	private static final double CURVE_SEGMENTS = 3; // Number of segments to approximate curves
	private static final double POINT_EPSILON = 0.0001; // Distance threshold for considering points equal

	/**
	 * Converts a JavaFX Text object into a list of cleaned vector lists
	 * representing the outlines
	 */
	public static List> extractOutlines(Path text) {
		List> rawOutlines = extractRawOutlines(text);
		List> cleanedOutlines = new ArrayList<>();

		for (List outline : rawOutlines) {
			List cleaned = cleanOutline(outline);
			if (cleaned.size() >= 3) { // Only keep outlines with at least 3 points
				cleanedOutlines.add(cleaned);
			}
		}

		return cleanedOutlines;
	}

	/**
	 * Initial extraction of outlines from text
	 */
	private static List> extractRawOutlines(Path textPath) {
		List> allOutlines = new ArrayList<>();

		List currentPath = new ArrayList<>();
		Vector3d lastPoint = null;

		for (PathElement element : textPath.getElements()) {
			if (element instanceof MoveTo) {
				if (!currentPath.isEmpty()) {
					allOutlines.add(new ArrayList<>(currentPath));
					currentPath.clear();
				}
				MoveTo move = (MoveTo) element;
				lastPoint = Vector3d.xyz(move.getX(), move.getY(), 0);
				currentPath.add(lastPoint);
			} else if (element instanceof LineTo) {
				LineTo line = (LineTo) element;
				lastPoint = Vector3d.xyz(line.getX(), line.getY(), 0);
				currentPath.add(lastPoint);
			} else if (element instanceof CubicCurveTo) {
				CubicCurveTo curve = (CubicCurveTo) element;
				List curvePoints = approximateCubicCurve(lastPoint,
						Vector3d.xyz(curve.getControlX1(), curve.getControlY1(), 0),
						Vector3d.xyz(curve.getControlX2(), curve.getControlY2(), 0),
						Vector3d.xyz(curve.getX(), curve.getY(), 0));
				currentPath.addAll(curvePoints);
				lastPoint = curvePoints.get(curvePoints.size() - 1);
			} else if (element instanceof QuadCurveTo) {
				QuadCurveTo curve = (QuadCurveTo) element;
				List curvePoints = approximateQuadCurve(lastPoint,
						Vector3d.xyz(curve.getControlX(), curve.getControlY(), 0),
						Vector3d.xyz(curve.getX(), curve.getY(), 0));
				currentPath.addAll(curvePoints);
				lastPoint = curvePoints.get(curvePoints.size() - 1);
			}
		}

		if (!currentPath.isEmpty()) {
			allOutlines.add(currentPath);
		}

		return allOutlines;
	}

	/**
	 * Clean an outline by removing duplicate points and ensuring proper closure
	 */
	private static List cleanOutline(List outline) {
		if (outline.isEmpty())
			return outline;

		List cleaned = new ArrayList<>();
		Vector3d prevPoint = null;

		// Process all points
		for (Vector3d point : outline) {
			if (prevPoint == null || !isNearlyEqual(prevPoint, point)) {
				// Only add point if it's significantly different from the previous point
				cleaned.add(point);
				prevPoint = point;
			}
		}
		// Remove redundant points that form zero-area triangles
		return removeRedundantPoints(cleaned);
	}

	/**
	 * Remove points that form zero-area triangles with their neighbors
	 */
	private static List removeRedundantPoints(List points) {
		if (points.size() < 3)
			return points;

		List result = new ArrayList<>();
		int size = points.size();

		for (int i = 0; i < size; i++) {
			Vector3d curr = points.get(i);
			result.add(curr);
		}

		return result;
	}

	private static boolean isNearlyEqual(Vector3d v1, Vector3d v2) {
		return v1.minus(v2).length() < POINT_EPSILON;
	}

	// Bezier curve methods remain the same
	private static List approximateCubicCurve(Vector3d start, Vector3d control1, Vector3d control2,
			Vector3d end) {
		List points = new ArrayList<>();
		for (double i = 1; i <= CURVE_SEGMENTS; i+=1) {
			double t = (double) i / CURVE_SEGMENTS;
			double x = cubicBezier(start.x, control1.x, control2.x, end.x, t);
			double y = cubicBezier(start.y, control1.y, control2.y, end.y, t);
			points.add(Vector3d.xyz(x, y, 0));
		}
		return points;
	}

	private static List approximateQuadCurve(Vector3d start, Vector3d control, Vector3d end) {
		List points = new ArrayList<>();
		for (double i = 1; i <= CURVE_SEGMENTS; i+=1) {
			double t = (double) i / CURVE_SEGMENTS;
			double x = quadBezier(start.x, control.x, end.x, t);
			double y = quadBezier(start.y, control.y, end.y, t);
			points.add(Vector3d.xyz(x, y, 0));
		}
		return points;
	}

	private static double cubicBezier(double p0, double p1, double p2, double p3, double t) {
		double mt = 1 - t;
		return p0 * mt * mt * mt + 3 * p1 * mt * mt * t + 3 * p2 * mt * t * t + p3 * t * t * t;
	}

	private static double quadBezier(double p0, double p1, double p2, double t) {
		double mt = 1 - t;
		return p0 * mt * mt + 2 * p1 * mt * t + p2 * t * t;
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy