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

com.googlecode.blaisemath.graphics.svg.SvgPathCoder Maven / Gradle / Ivy

package com.googlecode.blaisemath.graphics.svg;

/*-
 * #%L
 * blaise-graphics
 * --
 * Copyright (C) 2009 - 2024 Elisha Peterson
 * --
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *      http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * #L%
 */

import com.google.common.base.Strings;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import com.google.common.collect.Lists;
import com.google.common.primitives.Floats;
import com.googlecode.blaisemath.encode.StringDecoder;
import com.googlecode.blaisemath.encode.StringEncoder;

import java.awt.*;
import java.awt.geom.*;
import java.text.DecimalFormat;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;

import static com.google.common.base.Preconditions.checkArgument;

/**
 * Converts Java paths to/from SVG path strings.
 * @author Elisha Peterson
 */
public class SvgPathCoder implements StringEncoder, StringDecoder {

    private static final Logger LOG = Logger.getLogger(SvgPathCoder.class.getName());
    private static final DecimalFormat DF = new DecimalFormat("#.######");
    private static final DecimalFormat DF_LARGE = new DecimalFormat("#.######E0");

    // TODO - is this reasonable, given that Shape objects are not immutable?
    private static final LoadingCache PATH_CACHE = CacheBuilder.newBuilder()
            .maximumSize(1000)
            .build(new CacheLoader<>() {
                @Override
                public String load(Shape shape) {
                    return encodeIterator(shape.getPathIterator(null));
                }
            });

    @Override
    public String encode(Path2D obj) {
        return PATH_CACHE.getUnchecked(obj);
    }

    public String encodeShapePath(Shape obj) {
        return PATH_CACHE.getUnchecked(obj);
    }

    /**
     * Encode path iterator to an SVG path string.
     * @param pi iterator
     * @return path string
     */
    public static String encodeIterator(PathIterator pi) {
        float[] cur = new float[6];
        int curSegmentType;
        StringBuilder pathString = new StringBuilder();
        while (!pi.isDone()) {
            curSegmentType = pi.currentSegment(cur);
            switch (curSegmentType) {
                case PathIterator.SEG_MOVETO:
                    pathString.append("M ").append(numStr6(" ", cur[0], cur[1])).append(" ");
                    break;
                case PathIterator.SEG_LINETO:
                    pathString.append("L ").append(numStr6(" ", cur[0], cur[1])).append(" ");
                    break;
                case PathIterator.SEG_QUADTO:
                    pathString.append("Q ").append(numStr6(" ", cur[0], cur[1], cur[2], cur[3])).append(" ");
                    break;
                case PathIterator.SEG_CUBICTO:
                    pathString.append("C ").append(numStr6(" ", cur[0], cur[1], cur[2], cur[3], cur[4], cur[5])).append(" ");
                    break;
                case PathIterator.SEG_CLOSE:
                    pathString.append("Z");
                    break;
                default:
                    break;
            }
            pi.next();
        }
        return pathString.toString().trim();
    }

    @Override
    public Path2D decode(String svg) {
        if (Strings.isNullOrEmpty(svg)) {
            return new Path2D.Float();
        }
        String[] tokens = tokenizeSvgPath(svg);
        
        float[] start = {0f, 0f};
        float[] loc = {0f, 0f};
        float[] nextAnchor = {0f, 0f};

        Path2D.Float gp = new Path2D.Float();
        int pos = 0;
        while (pos < tokens.length) {
            SvgPathOperator op = operatorOf(tokens[pos].toLowerCase()); 
            float[] coordinates = nextFloats(tokens, pos+1);
            boolean relative = Character.isLowerCase(tokens[pos].charAt(0));
            op.apply(gp, coordinates, start, loc, nextAnchor, relative);
            pos += coordinates.length+1;
        }
        return gp;
    }
    
    //region STATIC UTILITIES

    /** Prints a sequence of numbers with the specified joiner and precision */
    static String numStr6(String join, double... vals) {
        if (vals.length == 0) {
            return "";
        }
        StringBuilder res = new StringBuilder();
        res.append(numStr6(vals[0]));
        for (int i = 1; i < vals.length; i++) {
            res.append(join).append(numStr6(vals[i]));
        }
        return res.toString();
    }

    /** Prints numbers w/ up to 6 digits of precision, removing trailing zeros */
    static String numStr6(double val) {
        return Math.abs(val) < 1E7 ? DF.format(val) : DF_LARGE.format(val);
    }
    
    /** Tokenize the path string, first making sure we have spaces in front of all numbers */
    private static String[] tokenizeSvgPath(String path) {
        return path.replaceAll("[A-Za-z]", " $0 ").trim()
                .replaceAll("[\\-]", " -")
                .split("[\\s,]+");
    }

    /** Get operator for given token */
    private static SvgPathOperator operatorOf(String token) {          
        checkArgument(token.length() == 1, "Invalid operator: "+token);
        for (SvgPathOperator op : SvgPathOperator.values()) {
            if (op.cmd == token.charAt(0)) {
                return op;
            }
        }
        checkArgument(false, "Invalid operator: "+token);
        return null;
    }
    
    /** 
     * Convert strings to floats, starting at given index, stopping at first non-float value
     * @param tokens array of strings
     * @param start starting index
     * @return coordinates extracted from string
     */
    private static float[] nextFloats(String[] tokens, int start) {
        List values = Lists.newArrayList();
        for (int i = start; i < tokens.length; i++) {
            try {
                values.add(Float.valueOf(tokens[i]));
            } catch (NumberFormatException ex) {
                LOG.log(Level.FINEST, "Not a float: " + tokens[i], ex);
                break;
            }
        }
        return Floats.toArray(values);
    }
    
    /** See http://stackoverflow.com/questions/1805101/svg-elliptical-arcs-with-java. */
    private static void arcTo(Path2D.Float path, float rx, float ry, float theta, boolean largeArcFlag, boolean sweepFlag, float x, float y) {
        // Ensure radii are valid
        if (rx == 0 || ry == 0) {
            path.lineTo(x, y);
            return;
        }
        // Get the current (x, y) coordinates of the path
        Point2D p2d = path.getCurrentPoint();
        float x0 = (float) p2d.getX();
        float y0 = (float) p2d.getY();
        // Compute the half distance between the current and the final point
        float dx2 = (x0 - x) / 2.0f;
        float dy2 = (y0 - y) / 2.0f;
        // Convert theta from degrees to radians
        theta = (float) Math.toRadians(theta % 360f);

        //
        // Step 1 : Compute (x1, y1)
        //
        float x1 = (float) (Math.cos(theta) * (double) dx2 + Math.sin(theta)
                        * (double) dy2);
        float y1 = (float) (-Math.sin(theta) * (double) dx2 + Math.cos(theta)
                        * (double) dy2);
        // Ensure radii are large enough
        rx = Math.abs(rx);
        ry = Math.abs(ry);
        float prx = rx * rx;
        float pry = ry * ry;
        float px1 = x1 * x1;
        float py1 = y1 * y1;
        double d = px1 / prx + py1 / pry;
        if (d > 1) {
            rx = Math.abs((float) (Math.sqrt(d) * (double) rx));
            ry = Math.abs((float) (Math.sqrt(d) * (double) ry));
            prx = rx * rx;
            pry = ry * ry;
        }

        //
        // Step 2 : Compute (cx1, cy1)
        //
        double sign = (largeArcFlag == sweepFlag) ? -1d : 1d;
        float coef = (float) (sign * Math .sqrt(((prx * pry) - (prx * py1) - (pry * px1))
                                        / ((prx * py1) + (pry * px1))));
        float cx1 = coef * ((rx * y1) / ry);
        float cy1 = coef * -((ry * x1) / rx);

        //
        // Step 3 : Compute (cx, cy) from (cx1, cy1)
        //
        float sx2 = (x0 + x) / 2.0f;
        float sy2 = (y0 + y) / 2.0f;
        float cx = sx2 + (float) (Math.cos(theta) * (double) cx1 - Math.sin(theta) * (double) cy1);
        float cy = sy2 + (float) (Math.sin(theta) * (double) cx1 + Math.cos(theta) * (double) cy1);

        //
        // Step 4 : Compute the angleStart (theta1) and the angleExtent (dtheta)
        //
        float ux = (x1 - cx1) / rx;
        float uy = (y1 - cy1) / ry;
        float vx = (-x1 - cx1) / rx;
        float vy = (-y1 - cy1) / ry;
        float p, n;
        // Compute the angle start
        n = (float) Math.sqrt((ux * ux) + (uy * uy));
        p = ux; // (1 * ux) + (0 * uy)
        sign = (uy < 0) ? -1d : 1d;
        float angleStart = (float) Math.toDegrees(sign * Math.acos(p / n));
        // Compute the angle extent
        n = (float) Math.sqrt((ux * ux + uy * uy) * (vx * vx + vy * vy));
        p = ux * vx + uy * vy;
        sign = (ux * vy - uy * vx < 0) ? -1d : 1d;
        float angleExtent = (float) Math.toDegrees(sign * Math.acos(p / n));
        if (!sweepFlag && angleExtent > 0) {
            angleExtent -= 360f;
        } else if (sweepFlag && angleExtent < 0) {
            angleExtent += 360f;
        }
        angleExtent %= 360f;
        angleStart %= 360f;

        Arc2D.Float arc = new Arc2D.Float();
        arc.x = cx - rx;
        arc.y = cy - ry;
        arc.width = rx * 2.0f;
        arc.height = ry * 2.0f;
        arc.start = -angleStart;
        arc.extent = -angleExtent;
        
        AffineTransform rotation = AffineTransform.getRotateInstance(theta, cx, cy);
        path.append(rotation.createTransformedShape(arc), true);
    }
    
    //endregion
    
    //region INNER CLASSES

    /** Helps decode based on operator type. */
    private enum SvgPathOperator {
        MOVE('m') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                for (int i = 0; i < coords.length-1; i+=2) {
                    loc[0] = relative ? loc[0]+coords[i] : coords[i];
                    loc[1] = relative ? loc[1]+coords[i+1] : coords[i+1];
                    if (i == 0) {
                        gp.moveTo(loc[0], loc[1]);
                    } else {
                        gp.lineTo(loc[0], loc[1]);
                    }
                }
                nextAnchor[0] = loc[0];
                nextAnchor[1] = loc[1];
                start[0] = loc[0];
                start[1] = loc[1];
            }
        },
        LINE('l') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                for (int i = 0; i < coords.length-1; i+=2) {
                    loc[0] = relative ? loc[0]+coords[i] : coords[i];
                    loc[1] = relative ? loc[1]+coords[i+1] : coords[i+1];
                    gp.lineTo(loc[0], loc[1]);
                }
                nextAnchor[0] = loc[0];
                nextAnchor[1] = loc[1];
            }
        },
        HLINE('h') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                for (float coord : coords) {
                    loc[0] = relative ? loc[0] + coord : coord;
                    gp.lineTo(loc[0], loc[1]);
                }
                nextAnchor[0] = loc[0];
                nextAnchor[1] = loc[1];
            }
        },
        VLINE('v') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                for (float coord : coords) {
                    loc[1] = relative ? loc[1] + coord : coord;
                    gp.lineTo(loc[0], loc[1]);
                }
                nextAnchor[0] = loc[0];
                nextAnchor[1] = loc[1];
            }
        },
        CUBIC_CURVE('c') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                for (int i = 0; i < coords.length-5; i+=6) {
                    float x1 = relative ? loc[0]+coords[i] : coords[i];
                    float y1 = relative ? loc[1]+coords[i+1] : coords[i+1];
                    float x2 = relative ? loc[0]+coords[i+2] : coords[i+2];
                    float y2 = relative ? loc[1]+coords[i+3] : coords[i+3];
                    loc[0] = relative ? loc[0]+coords[i+4] : coords[i+4];
                    loc[1] = relative ? loc[1]+coords[i+5] : coords[i+5];
                    gp.curveTo(x1, y1, x2, y2, loc[0], loc[1]);
                    nextAnchor[0] = 2*loc[0]-x2;
                    nextAnchor[1] = 2*loc[1]-y2;
                }
            }
        },
        SMOOTH_CURVE('s') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                for (int i = 0; i < coords.length-3; i+=4) {
                    float x1 = nextAnchor[0];
                    float y1 = nextAnchor[1];
                    float x2 = relative ? loc[0]+coords[i] : coords[i];
                    float y2 = relative ? loc[1]+coords[i+1] : coords[i+1];
                    loc[0] = relative ? loc[0]+coords[i+2] : coords[i+2];
                    loc[1] = relative ? loc[1]+coords[i+3] : coords[i+3];
                    gp.curveTo(x1, y1, x2, y2, loc[0], loc[1]);
                    nextAnchor[0] = 2*loc[0]-x2;
                    nextAnchor[1] = 2*loc[1]-y2;
                }
            }
        },
        QUAD_CURVE('q') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                for (int i = 0; i < coords.length-3; i+=4) {
                    float x1 = relative ? loc[0]+coords[i] : coords[i];
                    float y1 = relative ? loc[1]+coords[i+1] : coords[i+1];
                    loc[0] = relative ? loc[0]+coords[i+2] : coords[i+2];
                    loc[1] = relative ? loc[1]+coords[i+3] : coords[i+3];
                    gp.quadTo(x1, y1, loc[0], loc[1]);
                    nextAnchor[0] = 2*loc[0]-x1;
                    nextAnchor[1] = 2*loc[1]-y1;
                }
            }
        },
        SMOOTH_QUAD_CURVE('t') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                for (int i = 0; i < coords.length-1; i+=2) {
                    float x1 = nextAnchor[0];
                    float y1 = nextAnchor[1];
                    loc[0] = relative ? loc[0]+coords[i] : coords[i];
                    loc[1] = relative ? loc[1]+coords[i+1] : coords[i+1];
                    gp.quadTo(x1, y1, loc[0], loc[1]);
                    nextAnchor[0] = 2*loc[0]-x1;
                    nextAnchor[1] = 2*loc[1]-y1;
                }
            }
        },
        ARC('a') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                for (int i = 0; i < coords.length-6; i+=7) {
                    float rx = coords[i];
                    float ry = coords[i+1];
                    float xAxisRotation = coords[i+2];
                    boolean largeArcFlag = coords[i+3] == 1f;
                    boolean sweepFlag = coords[i+4] == 1f;
                    loc[0] = relative ? loc[0]+coords[i+5] : coords[i+5];
                    loc[1] = relative ? loc[1]+coords[i+6] : coords[i+6];
                    arcTo(gp, rx, ry, xAxisRotation, largeArcFlag, sweepFlag, loc[0], loc[1]);
                    nextAnchor[0] = loc[0];
                    nextAnchor[1] = loc[1];
                }
            }
        },
        CLOSE_PATH('z') {
            @Override
            void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative) {
                gp.closePath();
                loc[0] = start[0];
                loc[1] = start[1];
                nextAnchor[0] = start[0];
                nextAnchor[1] = start[1];
            }
        };
        
        private char cmd;

        SvgPathOperator(char cmd) {
            this.cmd = cmd;
        }

        /**
         * Apply the SVG command, adding the results onto the path.
         * @param gp path to add results onto
         * @param coords coordinates associated with the current command
         * @param start starting location for current subpath (modified by move commands)
         * @param loc current location (modified by most commands)
         * @param nextAnchor next anchor location (modified for curve commands)
         * @param relative whether coordinates are relative (true) or absolute (false)
         */
        abstract void apply(Path2D.Float gp, float[] coords, float[] start, float[] loc, float[] nextAnchor, boolean relative);
    }
    
    //endregion

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy