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

com.vividsolutions.jts.noding.InteriorIntersectionFinder Maven / Gradle / Ivy

The newest version!
/*
 * The JTS Topology Suite is a collection of Java classes that
 * implement the fundamental operations required to validate a given
 * geo-spatial data set to a known topological specification.
 *
 * Copyright (C) 2001 Vivid Solutions
 *
 * This library 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 2.1 of the License, or (at your option) any later version.
 *
 * This library 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 this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 *
 * For more information, contact:
 *
 *     Vivid Solutions
 *     Suite #1A
 *     2328 Government Street
 *     Victoria BC  V8T 5G5
 *     Canada
 *
 *     (250)385-6040
 *     www.vividsolutions.com
 */
package com.vividsolutions.jts.noding;

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

import com.vividsolutions.jts.geom.*;
import com.vividsolutions.jts.algorithm.LineIntersector;
//import com.vividsolutions.jts.util.Debug;

/**
 * Finds an interior intersection in a set of {@link SegmentString}s,
 * if one exists.  Only the first intersection found is reported.
 *
 * @version 1.7
 */
public class InteriorIntersectionFinder
    implements SegmentIntersector
{
	/**
	 * Creates an intersection finder which tests if there is at least one interior intersection.
	 * Uses short-circuiting for efficient performance.
	 * The intersection found is recorded.
	 * 
	 * @param li a line intersector
	 * @return a intersection finder which tests if there is at least one interior intersection.
	 */
	public static InteriorIntersectionFinder createAnyIntersectionFinder(LineIntersector li)
	{
		return new InteriorIntersectionFinder(li);
	}
	
	/**
	 * Creates an intersection finder which finds all interior intersections.
	 * The intersections are recorded for later inspection.
	 * 
	 * @param li a line intersector
	 * @return a intersection finder which finds all interior intersections.
	 */
	public static InteriorIntersectionFinder createAllIntersectionsFinder(LineIntersector li)
	{
		InteriorIntersectionFinder finder = new InteriorIntersectionFinder(li);
		finder.setFindAllIntersections(true);
		return finder;
	}
	
	/**
	 * Creates an intersection finder which counts all interior intersections.
	 * The intersections are note recorded to reduce memory usage.
	 * 
	 * @param li a line intersector
	 * @return a intersection finder which counts all interior intersections.
	 */
	public static InteriorIntersectionFinder createIntersectionCounter(LineIntersector li)
	{
		InteriorIntersectionFinder finder = new InteriorIntersectionFinder(li);
		finder.setFindAllIntersections(true);
		finder.setKeepIntersections(false);
		return finder;
	}
	
  private boolean findAllIntersections = false;
  private boolean isCheckEndSegmentsOnly = false;
  private LineIntersector li;
  private Coordinate interiorIntersection = null;
  private Coordinate[] intSegments = null;
  private List intersections = new ArrayList();
  private int intersectionCount = 0;
  private boolean keepIntersections = true;

  /**
   * Creates an intersection finder which finds an interior intersection
   * if one exists
   *
   * @param li the LineIntersector to use
   */
  public InteriorIntersectionFinder(LineIntersector li)
  {
    this.li = li;
    interiorIntersection = null;
  }

  /**
   * Sets whether all intersections should be computed.
   * When this is false (the default value)
   * the value of {@link #isDone()} is true after the first intersection is found.
   * 

* Default is false. * * @param findAllIntersections whether all intersections should be computed */ public void setFindAllIntersections(boolean findAllIntersections) { this.findAllIntersections = findAllIntersections; } /** * Sets whether intersection points are recorded. * If the only need is to count intersection points, this can be set to false. *

* Default is true. * * @param keepIntersections indicates whether intersections should be recorded */ public void setKeepIntersections(boolean keepIntersections) { this.keepIntersections = keepIntersections; } /** * Gets the intersections found. * * @return a List of {@link Coordinate) */ public List getIntersections() { return intersections; } /** * Gets the count of intersections found. * * @return the intersection count */ public int count() { return intersectionCount; } /** * Sets whether only end segments should be tested for interior intersection. * This is a performance optimization that may be used if * the segments have been previously noded by an appropriate algorithm. * It may be known that any potential noding failures will occur only in * end segments. * * @param isCheckEndSegmentsOnly whether to test only end segments */ public void setCheckEndSegmentsOnly(boolean isCheckEndSegmentsOnly) { this.isCheckEndSegmentsOnly = isCheckEndSegmentsOnly; } /** * Tests whether an intersection was found. * * @return true if an intersection was found */ public boolean hasIntersection() { return interiorIntersection != null; } /** * Gets the computed location of the intersection. * Due to round-off, the location may not be exact. * * @return the coordinate for the intersection location */ public Coordinate getInteriorIntersection() { return interiorIntersection; } /** * Gets the endpoints of the intersecting segments. * * @return an array of the segment endpoints (p00, p01, p10, p11) */ public Coordinate[] getIntersectionSegments() { return intSegments; } /** * This method is called by clients * of the {@link SegmentIntersector} class to process * intersections for two segments of the {@link SegmentString}s being intersected. * Note that some clients (such as MonotoneChains) may optimize away * this call for segment pairs which they have determined do not intersect * (e.g. by an disjoint envelope test). */ public void processIntersections( SegmentString e0, int segIndex0, SegmentString e1, int segIndex1 ) { // short-circuit if intersection already found if (! findAllIntersections && hasIntersection()) return; // don't bother intersecting a segment with itself if (e0 == e1 && segIndex0 == segIndex1) return; /** * If enabled, only test end segments (on either segString). * */ if (isCheckEndSegmentsOnly) { boolean isEndSegPresent = isEndSegment(e0, segIndex0) || isEndSegment(e1, segIndex1); if (! isEndSegPresent) return; } Coordinate p00 = e0.getCoordinates()[segIndex0]; Coordinate p01 = e0.getCoordinates()[segIndex0 + 1]; Coordinate p10 = e1.getCoordinates()[segIndex1]; Coordinate p11 = e1.getCoordinates()[segIndex1 + 1]; li.computeIntersection(p00, p01, p10, p11); //if (li.hasIntersection() && li.isProper()) Debug.println(li); if (li.hasIntersection()) { if (li.isInteriorIntersection()) { intSegments = new Coordinate[4]; intSegments[0] = p00; intSegments[1] = p01; intSegments[2] = p10; intSegments[3] = p11; interiorIntersection = li.getIntersection(0); if (keepIntersections) intersections.add(interiorIntersection); intersectionCount++; } } } /** * Tests whether a segment in a {@link SegmentString} is an end segment. * (either the first or last). * * @param segStr a segment string * @param index the index of a segment in the segment string * @return true if the segment is an end segment */ private boolean isEndSegment(SegmentString segStr, int index) { if (index == 0) return true; if (index >= segStr.size() - 2) return true; return false; } public boolean isDone() { if (findAllIntersections) return false; return interiorIntersection != null; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy