From eb43aa3de01f98820c878a4df5057b0889687426 Mon Sep 17 00:00:00 2001 From: Mike Barry Date: Fri, 27 May 2022 06:49:08 -0400 Subject: [PATCH 1/4] add --skip-filled-tiles option --- .../com/onthegomap/planetiler/VectorTile.java | 91 +++++++++++++++++++ .../planetiler/config/PlanetilerConfig.java | 6 +- .../planetiler/mbtiles/MbtilesWriter.java | 20 ++-- .../planetiler/PlanetilerTests.java | 19 ++++ .../onthegomap/planetiler/VectorTileTest.java | 51 +++++++++++ 5 files changed, 180 insertions(+), 7 deletions(-) diff --git a/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java b/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java index c4016d5e70..8586f319ac 100644 --- a/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java +++ b/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java @@ -76,6 +76,7 @@ public class VectorTile { private static final int EXTENT = 4096; private static final double SIZE = 256d; private final Map layers = new LinkedHashMap<>(); + private Boolean isFill = null; private static int[] getCommands(Geometry input, int scale) { var encoder = new CommandEncoder(scale); @@ -495,6 +496,23 @@ public byte[] encode() { return tile.build().toByteArray(); } + /** Returns true if this tile contains only polygon fills (for example, the middle of the ocean). */ + public boolean containsOnlyPolygonFills() { + if (isFill == null) { + boolean empty = true; + for (var layer : layers.values()) { + for (var feature : layer.encodedFeatures) { + empty = false; + if (!feature.geometry.isFill()) { + return false; + } + } + } + isFill = !empty; + } + return isFill; + } + private enum Command { MOVE_TO(1), LINE_TO(2), @@ -566,6 +584,79 @@ public String toString() { "], geomType=" + geomType + " (" + geomType.asByte() + ")]"; } + + /** Returns true if the encoded geometry is a polygon fill. */ + public boolean isFill() { + if (geomType != GeometryType.POLYGON) { + return false; + } + + int extent = EXTENT << scale; + int firstX = 0; + int firstY = 0; + int x = 0; + int y = 0; + + int geometryCount = commands.length; + int length = 0; + int command = 0; + int i = 0; + while (i < geometryCount) { + + if (length <= 0) { + length = commands[i++]; + command = length & ((1 << 3) - 1); + length = length >> 3; + } + + if (length > 0) { + if (command == Command.CLOSE_PATH.value) { + if (doesSegmentCrossTile(x, y, firstX, firstY, extent)) { + return false; + } + length--; + continue; + } + + int dx = commands[i++]; + int dy = commands[i++]; + + length--; + + dx = zigZagDecode(dx); + dy = zigZagDecode(dy); + + int nextX = x + dx; + int nextY = y + dy; + + if (command == Command.MOVE_TO.value) { + firstX = nextX; + firstY = nextY; + if (isInsideTile(firstX, firstY, extent)) { + return false; + } + } else if (doesSegmentCrossTile(x, y, nextX, nextY, extent)) { + return false; + } + y = nextY; + x = nextX; + } + + } + + return true; + } + + private static boolean isInsideTile(int x, int y, int extent) { + return x >= 0 && x <= extent && y >= 0 && y <= extent; + } + + private static boolean doesSegmentCrossTile(int x1, int y1, int x2, int y2, int extent) { + return (y1 >= 0 || y2 >= 0) && + (y1 <= extent || y2 <= extent) && + (x1 >= 0 || x2 >= 0) && + (x1 <= extent || x2 <= extent); + } } /** diff --git a/planetiler-core/src/main/java/com/onthegomap/planetiler/config/PlanetilerConfig.java b/planetiler-core/src/main/java/com/onthegomap/planetiler/config/PlanetilerConfig.java index 86286e9fa7..5b5bb96fad 100644 --- a/planetiler-core/src/main/java/com/onthegomap/planetiler/config/PlanetilerConfig.java +++ b/planetiler-core/src/main/java/com/onthegomap/planetiler/config/PlanetilerConfig.java @@ -41,7 +41,8 @@ public record PlanetilerConfig( double simplifyToleranceAtMaxZoom, double simplifyToleranceBelowMaxZoom, boolean osmLazyReads, - boolean compactDb + boolean compactDb, + boolean skipFilledTiles ) { public static final int MIN_MINZOOM = 0; @@ -142,6 +143,9 @@ public static PlanetilerConfig from(Arguments arguments) { false), arguments.getBoolean("compact_db", "Reduce the DB size by separating and deduping the tile data", + false), + arguments.getBoolean("skip_filled_tiles", + "Skip writing tiles containing only polygon fills to the output", false) ); } diff --git a/planetiler-core/src/main/java/com/onthegomap/planetiler/mbtiles/MbtilesWriter.java b/planetiler-core/src/main/java/com/onthegomap/planetiler/mbtiles/MbtilesWriter.java index bf17e1def7..1fd942b456 100644 --- a/planetiler-core/src/main/java/com/onthegomap/planetiler/mbtiles/MbtilesWriter.java +++ b/planetiler-core/src/main/java/com/onthegomap/planetiler/mbtiles/MbtilesWriter.java @@ -49,7 +49,6 @@ public class MbtilesWriter { private static final Logger LOGGER = LoggerFactory.getLogger(MbtilesWriter.class); private static final long MAX_FEATURES_PER_BATCH = 10_000; private static final long MAX_TILES_PER_BATCH = 1_000; - private static final int MAX_FEATURES_HASHING_THRESHOLD = 5; private final Counter.Readable featuresProcessed; private final Counter memoizedTiles; private final Mbtiles db; @@ -258,7 +257,9 @@ private void tileEncoder(Iterable prev, Consumer next) thr */ byte[] lastBytes = null, lastEncoded = null; Integer lastTileDataHash = null; + boolean lastIsFill = false; boolean compactDb = config.compactDb(); + boolean skipFilled = config.skipFilledTiles(); for (TileBatch batch : prev) { Queue result = new ArrayDeque<>(batch.size()); @@ -270,23 +271,30 @@ private void tileEncoder(Iterable prev, Consumer next) thr byte[] bytes, encoded; Integer tileDataHash; if (tileFeatures.hasSameContents(last)) { + if (skipFilled && lastIsFill) { + continue; + } bytes = lastBytes; encoded = lastEncoded; tileDataHash = lastTileDataHash; memoizedTiles.inc(); } else { VectorTile en = tileFeatures.getVectorTileEncoder(); - encoded = en.encode(); - bytes = gzip(encoded); + if (skipFilled) { + lastIsFill = en.containsOnlyPolygonFills(); + if (lastIsFill) { + continue; + } + } + lastEncoded = encoded = en.encode(); + lastBytes = bytes = gzip(encoded); last = tileFeatures; - lastEncoded = encoded; - lastBytes = bytes; if (encoded.length > 1_000_000) { LOGGER.warn("{} {}kb uncompressed", tileFeatures.tileCoord(), encoded.length / 1024); } - if (compactDb && tileFeatures.getNumFeaturesToEmit() < MAX_FEATURES_HASHING_THRESHOLD) { + if (compactDb && en.containsOnlyPolygonFills()) { tileDataHash = tileFeatures.generateContentHash(); } else { tileDataHash = null; diff --git a/planetiler-core/src/test/java/com/onthegomap/planetiler/PlanetilerTests.java b/planetiler-core/src/test/java/com/onthegomap/planetiler/PlanetilerTests.java index 60e82ea138..c41da48317 100644 --- a/planetiler-core/src/test/java/com/onthegomap/planetiler/PlanetilerTests.java +++ b/planetiler-core/src/test/java/com/onthegomap/planetiler/PlanetilerTests.java @@ -675,6 +675,25 @@ void testFullWorldPolygon() throws Exception { )).stream().map(d -> d.geometry().geom().norm()).toList()); } + @Test + void testSkipFill() throws Exception { + var results = runWithReaderFeatures( + Map.of("threads", "1", "skip-filled-tiles", "true"), + List.of( + newReaderFeature(WORLD_POLYGON, Map.of()) + ), + (in, features) -> features.polygon("layer") + .setZoomRange(0, 6) + .setBufferPixels(4) + ); + + assertEquals(481, results.tiles.size()); + // spot-check one filled tile does not exist + assertNull(results.tiles.get(TileCoord.ofXYZ( + Z4_TILES / 2, Z4_TILES / 2, 4 + ))); + } + @ParameterizedTest @CsvSource({ "chesapeake.wkb, 4076", diff --git a/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java b/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java index bde78c49a3..18020451d0 100644 --- a/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java +++ b/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java @@ -21,6 +21,7 @@ import static com.onthegomap.planetiler.TestUtils.*; import static com.onthegomap.planetiler.geo.GeoUtils.JTS_FACTORY; import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertFalse; import static org.junit.jupiter.api.Assertions.assertNotSame; import static org.junit.jupiter.api.Assertions.assertTrue; import static org.junit.jupiter.api.DynamicTest.dynamicTest; @@ -33,6 +34,8 @@ import org.junit.jupiter.api.DynamicTest; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.TestFactory; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.CsvSource; import org.locationtech.jts.geom.Coordinate; import org.locationtech.jts.geom.CoordinateXY; import org.locationtech.jts.geom.Geometry; @@ -333,6 +336,54 @@ void testMultipleFeaturesMultipleLayer() { assertEquals("layer2", decoded.get(2).layer()); } + @ParameterizedTest + @CsvSource({ + "true,-1,-1,257,257", + "true,-5,-5,260,260", + "false,-1,-1,254,254", + "false,0,-1,257,257", + "false,-1,0,257,257", + "false,-1,-1,256,257", + "false,-1,-1,257,256", + + "false,0,0,1,1", + "false,1,1,2,2", + "false,1,1,2,2", + }) + void testIsFill(boolean isFill, double x1, double y1, double x2, double y2) { + for (int scale = 0; scale < 4; scale++) { + assertEquals(isFill, VectorTile.encodeGeometry(rectangle(x1, y1, x2, y2), scale).isFill(), "scale=" + scale); + } + } + + @Test + void testCrossBoundaryNotFill() { + assertFalse(VectorTile.encodeGeometry(newPolygon( + -1, -1, + 257, -1, + -1, 257, + -1, -1 + ), 0).isFill()); + assertFalse(VectorTile.encodeGeometry(newPolygon( + 257, -1, + -1, 257, + -1, -1, + 257, -1 + ), 0).isFill()); + assertFalse(VectorTile.encodeGeometry(newPolygon( + -1, 257, + -1, -1, + 257, -1, + -1, 257 + ), 0).isFill()); + assertFalse(VectorTile.encodeGeometry(newPolygon( + -1, -1, + 513, -1, + -1, 513, + -1, -1 + ), 1).isFill()); + } + private void testRoundTripAttrs(Map attrs) { testRoundTrip(JTS_FACTORY.createPoint(new CoordinateXY(0, 0)), "layer", attrs, 1); } From 2d924c41e09eaf0a0e0727a767d141a121ce2138 Mon Sep 17 00:00:00 2001 From: Mike Barry Date: Fri, 27 May 2022 08:47:09 -0400 Subject: [PATCH 2/4] ensure feature is not just out of bounds --- .../com/onthegomap/planetiler/VectorTile.java | 54 +++++++++++++------ .../onthegomap/planetiler/VectorTileTest.java | 5 ++ 2 files changed, 43 insertions(+), 16 deletions(-) diff --git a/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java b/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java index 8586f319ac..0cb948049e 100644 --- a/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java +++ b/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java @@ -537,12 +537,41 @@ private enum Command { */ public record VectorGeometry(int[] commands, GeometryType geomType, int scale) { + private static final int LEFT = 1; + private static final int RIGHT = 1 << 1; + private static final int TOP = 1 << 2; + private static final int BOTTOM = 1 << 3; + private static final int INSIDE = 0; + private static final int ALL = TOP | LEFT | RIGHT | BOTTOM; + public VectorGeometry { if (scale < 0) { throw new IllegalArgumentException("scale can not be less than 0, got: " + scale); } } + private static int getSide(int x, int y, int extent) { + int result = INSIDE; + if (x < 0) { + result |= LEFT; + } else if (x > extent) { + result |= RIGHT; + } + if (y < 0) { + result |= TOP; + } else if (y > extent) { + result |= BOTTOM; + } + return result; + } + + private static boolean doesSegmentCrossTile(int x1, int y1, int x2, int y2, int extent) { + return (y1 >= 0 || y2 >= 0) && + (y1 <= extent || y2 <= extent) && + (x1 >= 0 || x2 >= 0) && + (x1 <= extent || x2 <= extent); + } + /** Converts an encoded geometry back to a JTS geometry. */ public Geometry decode() throws GeometryException { return decodeCommands(geomType, commands, scale); @@ -592,6 +621,7 @@ public boolean isFill() { } int extent = EXTENT << scale; + int visited = INSIDE; int firstX = 0; int firstY = 0; int x = 0; @@ -611,7 +641,7 @@ public boolean isFill() { if (length > 0) { if (command == Command.CLOSE_PATH.value) { - if (doesSegmentCrossTile(x, y, firstX, firstY, extent)) { + if (doesSegmentCrossTile(x, y, firstX, firstY, extent) || visited != ALL) { return false; } length--; @@ -632,11 +662,14 @@ public boolean isFill() { if (command == Command.MOVE_TO.value) { firstX = nextX; firstY = nextY; - if (isInsideTile(firstX, firstY, extent)) { + if ((visited = getSide(firstX, firstY, extent)) == INSIDE) { return false; } - } else if (doesSegmentCrossTile(x, y, nextX, nextY, extent)) { - return false; + } else { + if (doesSegmentCrossTile(x, y, nextX, nextY, extent)) { + return false; + } + visited |= getSide(nextX, nextY, extent); } y = nextY; x = nextX; @@ -644,18 +677,7 @@ public boolean isFill() { } - return true; - } - - private static boolean isInsideTile(int x, int y, int extent) { - return x >= 0 && x <= extent && y >= 0 && y <= extent; - } - - private static boolean doesSegmentCrossTile(int x1, int y1, int x2, int y2, int extent) { - return (y1 >= 0 || y2 >= 0) && - (y1 <= extent || y2 <= extent) && - (x1 >= 0 || x2 >= 0) && - (x1 <= extent || x2 <= extent); + return visited == ALL; } } diff --git a/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java b/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java index 18020451d0..cd143875f8 100644 --- a/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java +++ b/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java @@ -349,6 +349,11 @@ void testMultipleFeaturesMultipleLayer() { "false,0,0,1,1", "false,1,1,2,2", "false,1,1,2,2", + + "false,-10,-10,-5,-5", + "false,260,-10,270,5", + "false,-10,260,-5,270", + "false,260,260,270,270", }) void testIsFill(boolean isFill, double x1, double y1, double x2, double y2) { for (int scale = 0; scale < 4; scale++) { From eb4ca2aa367bebcb84cad5d175d8cb8c6d06c800 Mon Sep 17 00:00:00 2001 From: Mike Barry Date: Tue, 31 May 2022 20:45:01 -0400 Subject: [PATCH 3/4] cache is not fill --- .../src/main/java/com/onthegomap/planetiler/VectorTile.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java b/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java index 0cb948049e..b28d37e116 100644 --- a/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java +++ b/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java @@ -504,7 +504,7 @@ public boolean containsOnlyPolygonFills() { for (var feature : layer.encodedFeatures) { empty = false; if (!feature.geometry.isFill()) { - return false; + return isFill = false; } } } From 15029c2fc9ce25809d0a3feafe431151a07aaef3 Mon Sep 17 00:00:00 2001 From: Mike Barry Date: Fri, 3 Jun 2022 08:49:11 -0400 Subject: [PATCH 4/4] catch rectangle edges and horizontal/vertical lines --- .../com/onthegomap/planetiler/VectorTile.java | 91 +++++++++++++---- .../planetiler/mbtiles/MbtilesWriter.java | 4 +- .../onthegomap/planetiler/VectorTileTest.java | 99 +++++++++++++------ 3 files changed, 144 insertions(+), 50 deletions(-) diff --git a/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java b/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java index b28d37e116..9459eecec8 100644 --- a/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java +++ b/planetiler-core/src/main/java/com/onthegomap/planetiler/VectorTile.java @@ -76,7 +76,6 @@ public class VectorTile { private static final int EXTENT = 4096; private static final double SIZE = 256d; private final Map layers = new LinkedHashMap<>(); - private Boolean isFill = null; private static int[] getCommands(Geometry input, int scale) { var encoder = new CommandEncoder(scale); @@ -496,21 +495,32 @@ public byte[] encode() { return tile.build().toByteArray(); } - /** Returns true if this tile contains only polygon fills (for example, the middle of the ocean). */ - public boolean containsOnlyPolygonFills() { - if (isFill == null) { - boolean empty = true; - for (var layer : layers.values()) { - for (var feature : layer.encodedFeatures) { - empty = false; - if (!feature.geometry.isFill()) { - return isFill = false; - } + /** + * Returns true if this tile contains only polygon fills. + */ + public boolean containsOnlyFills() { + return containsOnlyFillsOrEdges(false); + } + + /** + * Returns true if this tile contains only polygon fills or horizontal/vertical edges that are likely to be repeated + * across tiles. + */ + public boolean containsOnlyFillsOrEdges() { + return containsOnlyFillsOrEdges(true); + } + + private boolean containsOnlyFillsOrEdges(boolean allowEdges) { + boolean empty = true; + for (var layer : layers.values()) { + for (var feature : layer.encodedFeatures) { + empty = false; + if (!feature.geometry.isFillOrEdge(allowEdges)) { + return false; } } - isFill = !empty; } - return isFill; + return !empty; } private enum Command { @@ -565,13 +575,35 @@ private static int getSide(int x, int y, int extent) { return result; } - private static boolean doesSegmentCrossTile(int x1, int y1, int x2, int y2, int extent) { + private static boolean slanted(int x1, int y1, int x2, int y2) { + return x1 != x2 && y1 != y2; + } + + private static boolean segmentCrossesTile(int x1, int y1, int x2, int y2, int extent) { return (y1 >= 0 || y2 >= 0) && (y1 <= extent || y2 <= extent) && (x1 >= 0 || x2 >= 0) && (x1 <= extent || x2 <= extent); } + private static boolean isSegmentInvalid(boolean allowEdges, int x1, int y1, int x2, int y2, int extent) { + boolean crossesTile = segmentCrossesTile(x1, y1, x2, y2, extent); + if (allowEdges) { + return crossesTile && slanted(x1, y1, x2, y2); + } else { + return crossesTile; + } + } + + + private static boolean visitedEnoughSides(boolean allowEdges, int sides) { + if (allowEdges) { + return ((sides & LEFT) > 0 && (sides & RIGHT) > 0) || ((sides & TOP) > 0 && (sides & BOTTOM) > 0); + } else { + return sides == ALL; + } + } + /** Converts an encoded geometry back to a JTS geometry. */ public Geometry decode() throws GeometryException { return decodeCommands(geomType, commands, scale); @@ -616,10 +648,28 @@ public String toString() { /** Returns true if the encoded geometry is a polygon fill. */ public boolean isFill() { - if (geomType != GeometryType.POLYGON) { + return isFillOrEdge(false); + } + + /** + * Returns true if the encoded geometry is a polygon fill, rectangle edge, or part of a horizontal/vertical line + * that is likely to be repeated across tiles. + */ + public boolean isFillOrEdge() { + return isFillOrEdge(true); + } + + /** + * Returns true if the encoded geometry is a polygon fill, or if {@code allowEdges == true} then also a rectangle + * edge, or part of a horizontal/vertical line that is likely to be repeated across tiles. + */ + public boolean isFillOrEdge(boolean allowEdges) { + if (geomType != GeometryType.POLYGON && (!allowEdges || geomType != GeometryType.LINE)) { return false; } + boolean isLine = geomType == GeometryType.LINE; + int extent = EXTENT << scale; int visited = INSIDE; int firstX = 0; @@ -637,11 +687,15 @@ public boolean isFill() { length = commands[i++]; command = length & ((1 << 3) - 1); length = length >> 3; + if (isLine && length > 2) { + return false; + } } if (length > 0) { if (command == Command.CLOSE_PATH.value) { - if (doesSegmentCrossTile(x, y, firstX, firstY, extent) || visited != ALL) { + if (isSegmentInvalid(allowEdges, x, y, firstX, firstY, extent) || + !visitedEnoughSides(allowEdges, visited)) { return false; } length--; @@ -666,7 +720,7 @@ public boolean isFill() { return false; } } else { - if (doesSegmentCrossTile(x, y, nextX, nextY, extent)) { + if (isSegmentInvalid(allowEdges, x, y, nextX, nextY, extent)) { return false; } visited |= getSide(nextX, nextY, extent); @@ -677,8 +731,9 @@ public boolean isFill() { } - return visited == ALL; + return visitedEnoughSides(allowEdges, visited); } + } /** diff --git a/planetiler-core/src/main/java/com/onthegomap/planetiler/mbtiles/MbtilesWriter.java b/planetiler-core/src/main/java/com/onthegomap/planetiler/mbtiles/MbtilesWriter.java index 1fd942b456..e5258d3bd5 100644 --- a/planetiler-core/src/main/java/com/onthegomap/planetiler/mbtiles/MbtilesWriter.java +++ b/planetiler-core/src/main/java/com/onthegomap/planetiler/mbtiles/MbtilesWriter.java @@ -281,7 +281,7 @@ private void tileEncoder(Iterable prev, Consumer next) thr } else { VectorTile en = tileFeatures.getVectorTileEncoder(); if (skipFilled) { - lastIsFill = en.containsOnlyPolygonFills(); + lastIsFill = en.containsOnlyFills(); if (lastIsFill) { continue; } @@ -294,7 +294,7 @@ private void tileEncoder(Iterable prev, Consumer next) thr tileFeatures.tileCoord(), encoded.length / 1024); } - if (compactDb && en.containsOnlyPolygonFills()) { + if (compactDb && en.containsOnlyFillsOrEdges()) { tileDataHash = tileFeatures.generateContentHash(); } else { tileDataHash = null; diff --git a/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java b/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java index cd143875f8..e8fe51caf8 100644 --- a/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java +++ b/planetiler-core/src/test/java/com/onthegomap/planetiler/VectorTileTest.java @@ -21,7 +21,6 @@ import static com.onthegomap.planetiler.TestUtils.*; import static com.onthegomap.planetiler.geo.GeoUtils.JTS_FACTORY; import static org.junit.jupiter.api.Assertions.assertEquals; -import static org.junit.jupiter.api.Assertions.assertFalse; import static org.junit.jupiter.api.Assertions.assertNotSame; import static org.junit.jupiter.api.Assertions.assertTrue; import static org.junit.jupiter.api.DynamicTest.dynamicTest; @@ -338,55 +337,95 @@ void testMultipleFeaturesMultipleLayer() { @ParameterizedTest @CsvSource({ - "true,-1,-1,257,257", - "true,-5,-5,260,260", - "false,-1,-1,254,254", - "false,0,-1,257,257", - "false,-1,0,257,257", - "false,-1,-1,256,257", - "false,-1,-1,257,256", - - "false,0,0,1,1", - "false,1,1,2,2", - "false,1,1,2,2", - - "false,-10,-10,-5,-5", - "false,260,-10,270,5", - "false,-10,260,-5,270", - "false,260,260,270,270", + "true,true,-1,-1,257,257", + "true,true,-5,-5,260,260", + "false,false,-1,-1,254,254", + "true,false,0,-1,257,257", + "true,false,-1,0,257,257", + "true,false,-1,-1,256,257", + "true,false,-1,-1,257,256", + + "false,false,0,0,1,1", + "false,false,1,1,2,2", + "false,false,1,1,2,2", + + "false,false,-10,-10,-5,-5", + "false,false,260,-10,270,5", + "false,false,-10,260,-5,270", + "false,false,260,260,270,270", + + "true,false,1,-1,257,257", + "true,false,1,-1,255,257", + "false,false,1,-1,255,255", }) - void testIsFill(boolean isFill, double x1, double y1, double x2, double y2) { - for (int scale = 0; scale < 4; scale++) { - assertEquals(isFill, VectorTile.encodeGeometry(rectangle(x1, y1, x2, y2), scale).isFill(), "scale=" + scale); - } + void testRectangleIsFillOrEdge(boolean isFillOrEdge, boolean isFill, double x1, double y1, double x2, double y2) { + assertIsFillOrEdge(isFillOrEdge, isFill, rectangle(x1, y1, x2, y2)); + } + + @Test + void testRectangleWithSlantedEdgeIsNotFill() { + assertIsFillOrEdge(false, false, newPolygon( + 1, -1, + 257, -1, + 257, 257, + 2, 257, + 1, -1 + )); + } + + @ParameterizedTest + @CsvSource({ + "true,1,-1,1,257", + "false,1,1,1,257", + "false,1,-1,1,255", + "false,1,-1,2,257" + }) + void testLineIsEdge(boolean isFillOrEdge, double x1, double y1, double x2, double y2) { + assertIsFillOrEdge(isFillOrEdge, false, newLineString( + x1, y1, + x2, y2 + )); } @Test - void testCrossBoundaryNotFill() { - assertFalse(VectorTile.encodeGeometry(newPolygon( + void testCrossBoundaryNotFillOrEdge() { + assertIsFillOrEdge(false, false, newPolygon( -1, -1, 257, -1, -1, 257, -1, -1 - ), 0).isFill()); - assertFalse(VectorTile.encodeGeometry(newPolygon( + )); + assertIsFillOrEdge(false, false, newPolygon( 257, -1, -1, 257, -1, -1, 257, -1 - ), 0).isFill()); - assertFalse(VectorTile.encodeGeometry(newPolygon( + )); + assertIsFillOrEdge(false, false, newPolygon( -1, 257, -1, -1, 257, -1, -1, 257 - ), 0).isFill()); - assertFalse(VectorTile.encodeGeometry(newPolygon( + )); + assertIsFillOrEdge(false, false, newPolygon( -1, -1, 513, -1, -1, 513, -1, -1 - ), 1).isFill()); + )); + } + + private static void assertIsFillOrEdge(boolean isFillOrEdge, boolean isFill, Geometry geom) { + for (int rotation : List.of(0, 90, 180, 270)) { + var rectangle = + AffineTransformation.rotationInstance(Math.PI * rotation / 180, 128, 128).transform(geom); + for (int scale = 0; scale < 4; scale++) { + assertEquals(isFillOrEdge, VectorTile.encodeGeometry(rectangle, scale).isFillOrEdge(), + "scale=" + scale + " rotation=" + rotation); + assertEquals(isFill, VectorTile.encodeGeometry(rectangle, scale).isFill(), + "scale=" + scale + " rotation=" + rotation); + } + } } private void testRoundTripAttrs(Map attrs) {