planetiler/src/main/java/com/onthegomap/flatmap/render/CoordinateSequenceExtractor...

126 wiersze
4.8 KiB
Java

package com.onthegomap.flatmap.render;
import com.onthegomap.flatmap.geo.GeoUtils;
import com.onthegomap.flatmap.geo.GeometryException;
import java.util.ArrayList;
import java.util.List;
import org.jetbrains.annotations.NotNull;
import org.locationtech.jts.algorithm.Area;
import org.locationtech.jts.geom.CoordinateSequence;
import org.locationtech.jts.geom.CoordinateSequences;
import org.locationtech.jts.geom.Geometry;
import org.locationtech.jts.geom.GeometryCollection;
import org.locationtech.jts.geom.LineString;
import org.locationtech.jts.geom.LinearRing;
import org.locationtech.jts.geom.Point;
import org.locationtech.jts.geom.Polygon;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
class CoordinateSequenceExtractor {
private static final Logger LOGGER = LoggerFactory.getLogger(CoordinateSequenceExtractor.class);
static List<List<CoordinateSequence>> extractGroups(Geometry geom, double minSize) {
List<List<CoordinateSequence>> result = new ArrayList<>();
extractGroups(geom, result, minSize);
return result;
}
private static void extractGroups(Geometry geom, List<List<CoordinateSequence>> groups, double minSize) {
if (geom.isEmpty()) {
// ignore
} else if (geom instanceof GeometryCollection) {
for (int i = 0; i < geom.getNumGeometries(); i++) {
extractGroups(geom.getGeometryN(i), groups, minSize);
}
} else if (geom instanceof Polygon polygon) {
extractGroupsFromPolygon(groups, minSize, polygon);
} else if (geom instanceof LinearRing linearRing) {
extractGroups(GeoUtils.JTS_FACTORY.createPolygon(linearRing), groups, minSize);
} else if (geom instanceof LineString lineString) {
if (lineString.getLength() >= minSize) {
groups.add(List.of(lineString.getCoordinateSequence()));
}
} else {
throw new RuntimeException("unrecognized geometry type: " + geom.getGeometryType());
}
}
private static void extractGroupsFromPolygon(List<List<CoordinateSequence>> groups, double minSize, Polygon polygon) {
CoordinateSequence outer = polygon.getExteriorRing().getCoordinateSequence();
double outerArea = Area.ofRingSigned(outer);
if (outerArea > 0) {
CoordinateSequences.reverse(outer);
}
if (Math.abs(outerArea) >= minSize) {
List<CoordinateSequence> group = new ArrayList<>(1 + polygon.getNumInteriorRing());
groups.add(group);
group.add(outer);
for (int i = 0; i < polygon.getNumInteriorRing(); i++) {
CoordinateSequence inner = polygon.getInteriorRingN(i).getCoordinateSequence();
double innerArea = Area.ofRingSigned(inner);
if (innerArea > 0) {
CoordinateSequences.reverse(inner);
}
if (Math.abs(innerArea) >= minSize) {
group.add(inner);
}
}
}
}
static Geometry reassembleLineStrings(List<List<CoordinateSequence>> geoms) {
List<LineString> lineStrings = new ArrayList<>();
for (List<CoordinateSequence> inner : geoms) {
for (CoordinateSequence coordinateSequence : inner) {
if (coordinateSequence.size() > 1) {
lineStrings.add(GeoUtils.JTS_FACTORY.createLineString(coordinateSequence));
}
}
}
return lineStrings.size() == 1 ? lineStrings.get(0) : GeoUtils.createMultiLineString(lineStrings);
}
@NotNull
static Geometry reassemblePolygons(List<List<CoordinateSequence>> groups) throws GeometryException {
int numGeoms = groups.size();
if (numGeoms == 1) {
return reassemblePolygon(groups.get(0));
} else {
Polygon[] polygons = new Polygon[numGeoms];
for (int i = 0; i < numGeoms; i++) {
polygons[i] = reassemblePolygon(groups.get(i));
}
return GeoUtils.JTS_FACTORY.createMultiPolygon(polygons);
}
}
private static Polygon reassemblePolygon(List<CoordinateSequence> group) throws GeometryException {
try {
LinearRing first = GeoUtils.JTS_FACTORY.createLinearRing(group.get(0));
LinearRing[] rest = new LinearRing[group.size() - 1];
for (int j = 1; j < group.size(); j++) {
CoordinateSequence seq = group.get(j);
CoordinateSequences.reverse(seq);
rest[j - 1] = GeoUtils.JTS_FACTORY.createLinearRing(seq);
}
return GeoUtils.JTS_FACTORY.createPolygon(first, rest);
} catch (IllegalArgumentException e) {
throw new GeometryException("Could not build polygon", e);
}
}
static Geometry reassemblePoints(List<List<CoordinateSequence>> result) {
List<Point> points = new ArrayList<>();
for (List<CoordinateSequence> inner : result) {
for (CoordinateSequence coordinateSequence : inner) {
if (coordinateSequence.size() == 1) {
points.add(GeoUtils.JTS_FACTORY.createPoint(coordinateSequence));
}
}
}
return points.size() == 1 ? points.get(0) : GeoUtils.createMultiPoint(points);
}
}