Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Stolstov/collection methods #178

Merged
merged 19 commits into from
Jun 21, 2018
Merged
Show file tree
Hide file tree
Changes from 9 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
package com.esri.core.geometry.ogc;

import com.esri.core.geometry.Envelope;
import com.esri.core.geometry.GeoJsonExportFlags;
import com.esri.core.geometry.Geometry;
import com.esri.core.geometry.GeometryCursor;
import com.esri.core.geometry.GeometryException;
Expand All @@ -35,15 +36,15 @@
import com.esri.core.geometry.OGCStructureInternal;
import com.esri.core.geometry.OperatorConvexHull;
import com.esri.core.geometry.OperatorDifference;
import com.esri.core.geometry.OperatorExportToGeoJson;
import com.esri.core.geometry.OperatorIntersection;
import com.esri.core.geometry.OperatorUnion;
import com.esri.core.geometry.Point;
import com.esri.core.geometry.Polygon;
import com.esri.core.geometry.Polyline;
import com.esri.core.geometry.SimpleGeometryCursor;
import com.esri.core.geometry.SpatialReference;
import com.esri.core.geometry.VertexDescription;
import com.esri.core.geometry.GeoJsonExportFlags;
import com.esri.core.geometry.OperatorExportToGeoJson;
import com.esri.core.geometry.OperatorUnion;
import com.esri.core.geometry.Point;

import java.nio.ByteBuffer;
import java.nio.ByteOrder;
Expand Down Expand Up @@ -546,9 +547,11 @@ public double distance(OGCGeometry another) {

double minD = Double.NaN;
for (int i = 0, n = numGeometries(); i < n; ++i) {
// TODO Skip expensive distance computation if bounding boxes are further away than minD
double d = geometryN(i).distance(another);
if (d < minD || Double.isNaN(minD)) {
minD = d;
// TODO Replace zero with tolerance defined by the spatial reference
if (minD == 0) {
break;
}
Expand All @@ -560,6 +563,29 @@ public double distance(OGCGeometry another) {

//
//Relational operations
@Override
public boolean overlaps(OGCGeometry another) {
//TODO
throw new UnsupportedOperationException();
}

@Override
public boolean touches(OGCGeometry another) {
//TODO
throw new UnsupportedOperationException();
}

@Override
public boolean crosses(OGCGeometry another) {
//TODO
throw new UnsupportedOperationException();
}

@Override
public boolean relate(OGCGeometry another, String matrix) {
throw new UnsupportedOperationException();
}

@Override
public boolean disjoint(OGCGeometry another) {
if (isEmpty() || another.isEmpty())
Expand Down Expand Up @@ -623,106 +649,110 @@ public boolean Equals(OGCGeometry another) {

OGCConcreteGeometryCollection gc1 = (OGCConcreteGeometryCollection)g1;
OGCConcreteGeometryCollection gc2 = (OGCConcreteGeometryCollection)g2;
// TODO Assuming input geometries are simple and valid, remove-overlaps would be a no-op.
// Hence, calling flatten() should be sufficient.
gc1 = gc1.flattenAndRemoveOverlaps();
gc2 = gc2.flattenAndRemoveOverlaps();
int n = gc1.numGeometries();
if (n != gc2.numGeometries()) {
return false;
}

boolean res = false;
for (int i = 0; i < n; ++i) {
if (!gc1.geometryN(i).Equals(gc2.geometryN(i))) {
return false;
}

res = true;
}

return res;
return n > 0;
}


private static OGCConcreteGeometryCollection toGeometryCollection(OGCGeometry geometry)
{
if (geometry.geometryType() != OGCConcreteGeometryCollection.TYPE) {
return new OGCConcreteGeometryCollection(geometry, geometry.getEsriSpatialReference());
}

return (OGCConcreteGeometryCollection) geometry;
}

private static List<Geometry> toList(GeometryCursor cursor)
{
List<Geometry> geometries = new ArrayList<Geometry>();
for (Geometry geometry = cursor.next(); geometry != null; geometry = cursor.next()) {
geometries.add(geometry);
}
return geometries;
}

//Topological
@Override
public OGCGeometry difference(OGCGeometry another) {
List<OGCConcreteGeometryCollection> list = wrapGeomsIntoList_(this, another);
list = prepare_for_ops_(list);
if (list.size() != 2) // this should not happen
throw new GeometryException("internal error");

if (isEmpty() || another.isEmpty()) {
return this;
}

List<Geometry> geometries = toList(prepare_for_ops_(toGeometryCollection(this)));
List<Geometry> otherGeometries = toList(prepare_for_ops_(toGeometryCollection(another)));

List<OGCGeometry> result = new ArrayList<OGCGeometry>();
OGCConcreteGeometryCollection coll1 = list.get(0);
OGCConcreteGeometryCollection coll2 = list.get(1);
for (int i = 0, n = coll1.numGeometries(); i < n; ++i) {
OGCGeometry cur = coll1.geometryN(i);
for (int j = 0, n2 = coll2.numGeometries(); j < n2; ++j) {
OGCGeometry geom2 = coll2.geometryN(j);
if (cur.dimension() > geom2.dimension())
for (Geometry geometry : geometries) {
for (Geometry otherGeometry : otherGeometries) {
if (geometry.getDimension() > otherGeometry.getDimension()) {
continue; //subtracting lower dimension has no effect.

cur = cur.difference(geom2);
if (cur.isEmpty())
}

geometry = OperatorDifference.local().execute(geometry, otherGeometry, esriSR, null);
if (geometry.isEmpty()) {
break;
}
}

if (!geometry.isEmpty()) {
result.add(OGCGeometry.createFromEsriGeometry(geometry, esriSR));
}

if (cur.isEmpty())
continue;

result.add(cur);
}

if (result.size() == 1) {
result.get(0).reduceFromMulti();
return result.get(0).reduceFromMulti();
}

return new OGCConcreteGeometryCollection(result, esriSR);
return new OGCConcreteGeometryCollection(result, esriSR).flattenAndRemoveOverlaps();
}

@Override
public OGCGeometry intersection(OGCGeometry another) {
List<OGCConcreteGeometryCollection> list = wrapGeomsIntoList_(this, another);
list = prepare_for_ops_(list);
if (list.size() != 2) // this should not happen
throw new GeometryException("internal error");

if (isEmpty() || another.isEmpty()) {
return new OGCConcreteGeometryCollection(esriSR);
}

List<Geometry> geometries = toList(prepare_for_ops_(toGeometryCollection(this)));
List<Geometry> otherGeometries = toList(prepare_for_ops_(toGeometryCollection(another)));

List<OGCGeometry> result = new ArrayList<OGCGeometry>();
OGCConcreteGeometryCollection coll1 = list.get(0);
OGCConcreteGeometryCollection coll2 = list.get(1);
for (int i = 0, n = coll1.numGeometries(); i < n; ++i) {
OGCGeometry cur = coll1.geometryN(i);
for (int j = 0, n2 = coll2.numGeometries(); j < n2; ++j) {
OGCGeometry geom2 = coll2.geometryN(j);

OGCGeometry partialIntersection = cur.intersection(geom2);
if (partialIntersection.isEmpty())
continue;

result.add(partialIntersection);
for (Geometry geometry : geometries) {
for (Geometry otherGeometry : otherGeometries) {
GeometryCursor intersectionCursor = OperatorIntersection.local().execute(new SimpleGeometryCursor(geometry), new SimpleGeometryCursor(otherGeometry), esriSR, null, 7);
OGCGeometry intersection = OGCGeometry.createFromEsriCursor(intersectionCursor, esriSR, true);
if (!intersection.isEmpty()) {
result.add(intersection);
}
}
}

if (result.size() == 1) {
result.get(0).reduceFromMulti();
return result.get(0).reduceFromMulti();
}

return (new OGCConcreteGeometryCollection(result, esriSR)).flattenAndRemoveOverlaps();
return new OGCConcreteGeometryCollection(result, esriSR).flattenAndRemoveOverlaps();
}

//make a list of collections out of two geometries
private static List<OGCConcreteGeometryCollection> wrapGeomsIntoList_(OGCGeometry g1, OGCGeometry g2) {
List<OGCConcreteGeometryCollection> list = new ArrayList<OGCConcreteGeometryCollection>();
if (g1.geometryType() != OGCConcreteGeometryCollection.TYPE) {
g1 = new OGCConcreteGeometryCollection(g1, g1.getEsriSpatialReference());
}
if (g2.geometryType() != OGCConcreteGeometryCollection.TYPE) {
g2 = new OGCConcreteGeometryCollection(g2, g2.getEsriSpatialReference());
}

list.add((OGCConcreteGeometryCollection)g1);
list.add((OGCConcreteGeometryCollection)g2);

return list;

@Override
public OGCGeometry symDifference(OGCGeometry another) {
//TODO
throw new UnsupportedOperationException();
}

/**
* Checks if collection is flattened.
* @return True for the flattened collection. A flattened collection contains up to three non-empty geometries:
Expand Down Expand Up @@ -834,24 +864,23 @@ public OGCConcreteGeometryCollection flatten() {
/**
* Fixes topological overlaps in the GeometryCollecion.
* This is equivalent to union of the geometry collection elements.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

flatten method already performs union on polygons; should flatten also union points and line strings?

*
*
* TODO "flattened" collection is supposed to contain only mutli-geometries, but this method may return single geometries
* e.g. for GEOMETRYCOLLECTION (LINESTRING (...)) it returns GEOMETRYCOLLECTION (LINESTRING (...))
* and not GEOMETRYCOLLECTION (MULTILINESTRING (...))
* @return A geometry collection that is flattened and has no overlapping elements.
*/
public OGCConcreteGeometryCollection flattenAndRemoveOverlaps() {
ArrayList<Geometry> geoms = new ArrayList<Geometry>();


//flatten and crack/cluster
GeometryCursor cursor = OGCStructureInternal.prepare_for_ops_(flatten().getEsriGeometryCursor(), esriSR);
for (Geometry g = cursor.next(); g != null; g = cursor.next()) {
geoms.add(g);
}


//make sure geometries don't overlap
return removeOverlapsHelper_(geoms);
return new OGCConcreteGeometryCollection(removeOverlapsHelper_(toList(cursor)), esriSR);
}
private OGCConcreteGeometryCollection removeOverlapsHelper_(List<Geometry> geoms) {
ArrayList<Geometry> result = new ArrayList<Geometry>();

private GeometryCursor removeOverlapsHelper_(List<Geometry> geoms) {
List<Geometry> result = new ArrayList<Geometry>();
for (int i = 0; i < geoms.size(); ++i) {
Geometry current = geoms.get(i);
if (current.isEmpty())
Expand All @@ -870,7 +899,7 @@ private OGCConcreteGeometryCollection removeOverlapsHelper_(List<Geometry> geoms
result.add(current);
}

return new OGCConcreteGeometryCollection(new SimpleGeometryCursor(result), esriSR);
return new SimpleGeometryCursor(result);
}

private static class FlatteningCollectionCursor extends GeometryCursor {
Expand Down Expand Up @@ -921,36 +950,9 @@ public int getGeometryID() {
//Collectively processes group of geometry collections (intersects all segments and clusters points).
//Flattens collections, removes overlaps.
//Once done, the result collections would work well for topological and relational operations.
private List<OGCConcreteGeometryCollection> prepare_for_ops_(List<OGCConcreteGeometryCollection> geoms) {
assert(geoms != null && !geoms.isEmpty());
GeometryCursor prepared = OGCStructureInternal.prepare_for_ops_(new FlatteningCollectionCursor(geoms), esriSR);

List<OGCConcreteGeometryCollection> result = new ArrayList<OGCConcreteGeometryCollection>();
int prevCollectionIndex = -1;
List<Geometry> list = null;
for (Geometry g = prepared.next(); g != null; g = prepared.next()) {
int c = prepared.getGeometryID();
if (c != prevCollectionIndex) {
//add empty collections for all skipped indices
for (int i = prevCollectionIndex; i < c - 1; i++) {
result.add(new OGCConcreteGeometryCollection(esriSR));
}

if (list != null) {
result.add(removeOverlapsHelper_(list));
}

list = new ArrayList<Geometry>();
prevCollectionIndex = c;
}

list.add(g);
}

if (list != null) {
result.add(removeOverlapsHelper_(list));
}

return result;
private GeometryCursor prepare_for_ops_(OGCConcreteGeometryCollection collection) {
assert(collection != null && !collection.isEmpty());
GeometryCursor prepared = OGCStructureInternal.prepare_for_ops_(collection.flatten().getEsriGeometryCursor(), esriSR);
return removeOverlapsHelper_(toList(prepared));
}
}
Loading