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

org.locationtech.jts.io.geojson.OrientationTransformer Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2021 Martin Davis.
 *
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License 2.0
 * and Eclipse Distribution License v. 1.0 which accompanies this distribution.
 * The Eclipse Public License is available at http://www.eclipse.org/legal/epl-v20.html
 * and the Eclipse Distribution License is available at
 *
 * http://www.eclipse.org/org/documents/edl-v10.php.
 */
package org.locationtech.jts.io.geojson;

import org.locationtech.jts.algorithm.Orientation;
import org.locationtech.jts.geom.Geometry;
import org.locationtech.jts.geom.GeometryFactory;
import org.locationtech.jts.geom.LinearRing;
import org.locationtech.jts.geom.MultiPolygon;
import org.locationtech.jts.geom.Polygon;

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

/**
 * Utilities to modify the ring orientation of polygonal geometries.
 */
public class OrientationTransformer {

    /**
     * Transforms a geometry using the Right Hand Rule specifications defined
     * in the latest GeoJSON specification.
     * See RFC-7946 Specification for more context.
     *
     * @param geometry to be transformed
     * @return Geometry under the Right Hand Rule specifications
     */
    public static Geometry transformCCW(final Geometry geometry) {

        if (geometry instanceof MultiPolygon) {
            MultiPolygon multiPolygon = (MultiPolygon) geometry;

            List polygons = new ArrayList<>();
            for (int i = 0; i < multiPolygon.getNumGeometries(); i++) {
                final Geometry polygon = multiPolygon.getGeometryN(i);
                polygons.add((Polygon) transformCCW(polygon));
            }

            return new GeometryFactory().createMultiPolygon(polygons.toArray(new Polygon[0]));

        } else if (geometry instanceof Polygon) {
            return transformCCW((Polygon) geometry);

        } else {
            return geometry;
        }
    }

    /**
     * Transforms a polygon using the Right Hand Rule specifications defined
     * in the latest GeoJSON specification.
     * See RFC-7946 Specification for more context.
     *
     * @param polygon to be transformed
     * @return Polygon under the Right Hand Rule specifications
     */
    public static Polygon transformCCW(Polygon polygon) {
        LinearRing exteriorRing = polygon.getExteriorRing();
        LinearRing exteriorRingEnforced = transformCCW(exteriorRing, true);

        List interiorRings = new ArrayList<>();
        for (int i = 0; i < polygon.getNumInteriorRing(); i++) {
            interiorRings.add(transformCCW(polygon.getInteriorRingN(i), false));
        }

        return new GeometryFactory(polygon.getPrecisionModel(), polygon.getSRID())
                .createPolygon(exteriorRingEnforced, interiorRings.toArray(new LinearRing[0]));
    }

    /**
     * Transforms a polygon using the Right Hand Rule specifications defined
     * in the latest GeoJSON specification.
     * A linear ring MUST follow the right-hand rule with respect to the
     * area it bounds, i.e., exterior rings are counterclockwise, and
     * holes are clockwise.
     *
     * See RFC 7946 Specification for more context.
     *
     * @param ring the LinearRing, a constraint specific to Polygons
     * @param isExteriorRing true if the LinearRing is the exterior polygon ring, the one that defines the boundary
     * @return LinearRing under the Right Hand Rule specifications
     */
    public static LinearRing transformCCW(LinearRing ring, boolean isExteriorRing) {
        final boolean isRingClockWise = !Orientation.isCCW(ring.getCoordinateSequence());

        final LinearRing rightHandRuleRing;
        if (isExteriorRing) {
            rightHandRuleRing = isRingClockWise? ring.reverse() : (LinearRing) ring.copy();
        } else {
            rightHandRuleRing = isRingClockWise? (LinearRing) ring.copy() : ring.reverse();
        }
        return rightHandRuleRing;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy