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

org.geolatte.geom.Polygon Maven / Gradle / Ivy

/*
 * This file is part of the GeoLatte project.
 *
 *     GeoLatte is free software: you can redistribute it and/or modify
 *     it under the terms of the GNU Lesser General Public License as published by
 *     the Free Software Foundation, either version 3 of the License, or
 *     (at your option) any later version.
 *
 *     GeoLatte is distributed in the hope that it will be useful,
 *     but WITHOUT ANY WARRANTY; without even the implied warranty of
 *     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *     GNU Lesser General Public License for more details.
 *
 *     You should have received a copy of the GNU Lesser General Public License
 *     along with GeoLatte.  If not, see .
 *
 * Copyright (C) 2010 - 2011 and Ownership of code is shared by:
 * Qmino bvba - Romeinsestraat 18 - 3001 Heverlee  (http://www.qmino.com)
 * Geovise bvba - Generaal Eisenhowerlei 9 - 2140 Antwerpen (http://www.geovise.com)
 */

package org.geolatte.geom;

import org.geolatte.geom.crs.CoordinateReferenceSystem;

import java.util.Arrays;
import java.util.Iterator;

/**
 * A planar surface defined by 1 exterior boundary and 0 or more interior boundaries. Each interior boundary defines a
 * hole in the Polygon.
 *
 * 

The exterior boundary LinearRing defines the "top" of the surface which is the side of the surface * from which the exterior boundary appears to traverse the boundary in a counter clockwise direction. The interior * LinearRings will have the opposite orientation and appear as clockwise when viewed from the "top".

* *

The rules that define valid Polygons are as follows

*
    *
  1. Polygons are topologically closed;
  2. *
  3. The boundary of a Polygon consists of a set of LinearRings that make up its * exterior and interior boundaries;
  4. *
  5. No two rings in the boundary cross and the rings in the boundary of a Polygon may * intersect at a Point but only as a tangent
  6. *
  7. A Polygon may not have cut lines, spikes or punctures;
  8. *
  9. The interior of every Polygon is a connected point set;
  10. *
  11. The exterior of a Polygon with 1 or more holes is not connected. Each hole defines a * connected component of the exterior
  12. *
* * @author Karel Maesen, Geovise BVBA * creation-date: 4/14/11 */ public class Polygon

extends Geometry

implements Polygonal

, Complex> { private final LinearRing

[] rings; @SuppressWarnings("unchecked") public Polygon(CoordinateReferenceSystem

crs) { super(crs); rings = (LinearRing

[])new LinearRing[0]; } /** * Creates a Polygon with no holes, and having the specified PositionSequence as exterior boundary * * @param positionSequence the PositionSequence representing the exterior boundary * @throws IllegalArgumentException when the specified PositionSequence does not form a * LinearRing (i.e., is empty or not closed). */ @SuppressWarnings("unchecked") public Polygon(PositionSequence

positionSequence, CoordinateReferenceSystem

crs) { this(new LinearRing

(positionSequence, crs)); } /** * Creates a Polygon with the specified array of exterior and interior boundaries. * * @param rings the array of the Polygon's boundaries: the first element is the exterior boundary, all subsequent rings represent * interior boundaries. * @throws IllegalArgumentException when the specified PositionSequence does not form a * LinearRing (i.e., is empty or not closed). */ public Polygon(LinearRing

... rings) { super(nestPositionSequences(rings), getCrs(rings)); checkRings(rings); this.rings = Arrays.copyOf(rings, rings.length); } private void checkRings(LinearRing

[] rings) { CoordinateReferenceSystem

crs = getCrs(rings); for (LinearRing

ring : rings) { checkLinearRing(ring, crs); } } private void checkLinearRing(LinearRing

ring, CoordinateReferenceSystem

crs) { if (ring == null) throw new IllegalArgumentException("NULL linear ring is not valid."); if (ring.isEmpty()) throw new IllegalArgumentException("Empty linear ring is not valid."); if (!ring.getCoordinateReferenceSystem().equals(crs)) throw new IllegalArgumentException("Linear ring with different CRS than exterior boundary."); } /** * Returns the exterior boundary of this Polygon. * * @return a LinearRing representing the exterior boundary of this Polygon. */ public LinearRing

getExteriorRing() { return this.isEmpty() ? new LinearRing

(getPositions(), getCoordinateReferenceSystem()) : this.rings[0]; } /** * returns the number of interior boundaries. * * @return the number of interior boundaries */ public int getNumInteriorRing() { return this.isEmpty() ? 0 : this.rings.length - 1; } /** * Returns the specified interior ring. * * @param index the (zero-based) position of the interior boundary in the list of interior boundaries. * @return the LinearRing at the position specified by the parameter index. */ public LinearRing

getInteriorRingN(int index) { return this.rings[index + 1]; } @Override public int getDimension() { return 2; } @Override public GeometryType getGeometryType() { return GeometryType.POLYGON; } /** * Returns an Iterator over the boundaries of this Polygon. * *

The boundaries are returned in order, with the first element being the exterior boundary.

* * @return an Iterator over the boundaries of this Polygon. */ public Iterator> iterator() { return new Iterator>() { private int index = 0; @Override public boolean hasNext() { return index < rings.length; } @Override public LinearRing

next() { return rings[index++]; } @Override public void remove() { throw new UnsupportedOperationException(); } }; } @Override public void accept(GeometryVisitor

visitor) { visitor.visit(this); //TODO visit the individual rings } @Override public int getNumGeometries() { return rings.length; } @Override public Class getComponentType() { return LinearRing.class; } /** * Returns the components * * @return an array containing all component objects */ @Override public LinearRing

[] components() { return Arrays.copyOf(this.rings, this.rings.length); } @Override @SuppressWarnings("unchecked") public Polygon as(Class castToType){ checkCast(castToType); return (Polygon)this; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy