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.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;

// 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;
	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) {
		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)));
		// Convert Path elements into lists of points defining the perimeter
		// (exterior or interior)
		subtract.getElements().forEach(this::getPoints);

		// Group exterior polygons with their interior polygons
//		polis.stream().filter(LineSegment::isHole).forEach(hole -> {
//			polis.stream().filter(poly -> !poly.isHole())
//					.filter(poly -> !((Path) Shape.intersect(poly.getPath(), hole.getPath())).getElements().isEmpty())
//					.filter(poly -> poly.getPath().contains(new Point2D(hole.getOrigen().x, hole.getOrigen().y)))
//					.forEach(poly -> poly.addHole(hole));
//		});
		//polis.removeIf(LineSegment::isHole);
		
		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());
	    }
	    
	    private void getPoints(PathElement elem){
	        if(elem instanceof MoveTo){
	        	loadPoints();
	            p0=new Vector3d((float)((MoveTo)elem).getX(),(float)((MoveTo)elem).getY(),0f);
	            points.add(p0);
	        } else if(elem instanceof LineTo){
	            points.add(new Vector3d((float)((LineTo)elem).getX(),(float)((LineTo)elem).getY(),0f));
	        } else if(elem instanceof CubicCurveTo){
	            Vector3d ini = (points.size()>0?points.get(points.size()-1):p0);
	            IntStream.rangeClosed(1, POINTS_CURVE).forEach(i->points.add(evalCubicBezier((CubicCurveTo)elem, ini, ((double)i)/POINTS_CURVE)));
	        } else if(elem instanceof QuadCurveTo){
	            Vector3d ini = (points.size()>0?points.get(points.size()-1):p0);
	            IntStream.rangeClosed(1, POINTS_CURVE).forEach(i->points.add(evalQuadBezier((QuadCurveTo)elem, ini, ((double)i)/POINTS_CURVE)));
	        } else if(elem instanceof ClosePath){
	            points.add(p0);
	            // Every closed path is a polygon (exterior or interior==hole)
	            // the text, the list of points and a new path between them are
	            // stored in a LineSegment: a continuous line that can change direction
	            if(Math.abs(getArea())>0.001){
	                LineSegment line = new LineSegment(text);
	                line.setHole(isHole());
	                line.setPoints(points);
	                line.setPath(generatePath());
	                line.setOrigen(p0);
	                polis.add(line);
	            }
	            loadPoints();
	            
	        } 
	    }
	    
	    private void loadPoints(){
        	if(points.size()>4){
	        	points.remove(points.size() - 1);
				//points.remove(points.size() - 1);
				boolean hole = Extrude.isCCW(Polygon.fromPoints(points));
				CSG newLetter = Extrude.points(new Vector3d(0, 0, dir), points);

				if (!hole)
					sections.add(newLetter);
				else
					holes.add(newLetter);
        	}
            points=new ArrayList<>();
	    }
	    
	    private Vector3d evalCubicBezier(CubicCurveTo c, Vector3d ini, double t){
	        Vector3d p=new Vector3d((float)(Math.pow(1-t,3)*ini.x+
	                3*t*Math.pow(1-t,2)*c.getControlX1()+
	                3*(1-t)*t*t*c.getControlX2()+
	                Math.pow(t, 3)*c.getX()),
	                (float)(Math.pow(1-t,3)*ini.y+
	                3*t*Math.pow(1-t, 2)*c.getControlY1()+
	                3*(1-t)*t*t*c.getControlY2()+
	                Math.pow(t, 3)*c.getY()),
	                0f);
	        return p;
	    }
	    
	    private Vector3d evalQuadBezier(QuadCurveTo c, Vector3d ini, double t){
	        Vector3d p=new Vector3d((float)(Math.pow(1-t,2)*ini.x+
	                2*(1-t)*t*c.getControlX()+
	                Math.pow(t, 2)*c.getX()),
	                (float)(Math.pow(1-t,2)*ini.y+
	                2*(1-t)*t*c.getControlY()+
	                Math.pow(t, 2)*c.getY()),
	                0f);
	        return p;
	    }
	    
	    private double getArea(){
	        DoubleProperty res=new SimpleDoubleProperty();
	        IntStream.range(0, points.size()-1)
	                .forEach(i->res.set(res.get()+points.get(i).cross(points.get(i+1)).z));
	        // System.out.println("path: "+res.doubleValue()/2);
	        
	        return res.doubleValue()/2d;
	    }
	    
	    private boolean isHole(){
	        // area>0 -> the path is a hole, clockwise (y up)
	        // area<0 -> the path is a polygon, counterclockwise (y up)
	        return getArea()>0;
	    }
	    
	    private Path generatePath(){
	        Path path = new Path(new MoveTo(points.get(0).x,points.get(0).y));
	        points.stream().skip(1).forEach(p->path.getElements().add(new LineTo(p.x,p.y)));
	        path.getElements().add(new ClosePath());
	        path.setStroke(Color.GREEN);
	        // Path must be filled to allow Shape.intersect
	        path.setFill(Color.RED);
	        return path;
	    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy