From e83b7e40aa0d8410b76aebb0f588c5800df5b606 Mon Sep 17 00:00:00 2001 From: Martin Davis Date: Tue, 7 Jan 2025 11:40:41 -0800 Subject: [PATCH] Fix OffsetCurve to support mitre joins for polygons (#1109) --- .../jts/operation/buffer/OffsetCurve.java | 40 ++++++++++++------- .../jts/operation/buffer/OffsetCurveTest.java | 15 +++++++ 2 files changed, 40 insertions(+), 15 deletions(-) diff --git a/modules/core/src/main/java/org/locationtech/jts/operation/buffer/OffsetCurve.java b/modules/core/src/main/java/org/locationtech/jts/operation/buffer/OffsetCurve.java index 25d044cccd..404c47bd36 100644 --- a/modules/core/src/main/java/org/locationtech/jts/operation/buffer/OffsetCurve.java +++ b/modules/core/src/main/java/org/locationtech/jts/operation/buffer/OffsetCurve.java @@ -208,27 +208,37 @@ public Geometry getCurve() { public Geometry map(Geometry geom) { if (geom instanceof Point) return null; if (geom instanceof Polygon ) { - return toLineString(geom.buffer(distance).getBoundary()); + return computePolygonCurve((Polygon) geom, distance); } return computeCurve((LineString) geom, distance); } - - /** - * Force LinearRings to be LineStrings. - * - * @param geom a geometry which may be a LinearRing - * @return a geometry which will be a LineString or MultiLineString - */ - private Geometry toLineString(Geometry geom) { - if (geom instanceof LinearRing) { - LinearRing ring = (LinearRing) geom; - return geom.getFactory().createLineString(ring.getCoordinateSequence()); - } - return geom; - } }); } + private Geometry computePolygonCurve(Polygon poly, double distance) { + Geometry buffer; + if (bufferParams == null) + buffer = BufferOp.bufferOp(poly, distance); + else { + buffer = BufferOp.bufferOp(poly, distance, bufferParams); + } + return toLineString(buffer.getBoundary()); + } + + /** + * Force LinearRings to be LineStrings. + * + * @param geom a linear geometry which may be a LinearRing + * @return a geometry which if linear is a LineString or MultiLineString + */ + private static Geometry toLineString(Geometry geom) { + if (geom instanceof LinearRing) { + LinearRing ring = (LinearRing) geom; + return geom.getFactory().createLineString(ring.getCoordinateSequence()); + } + return geom; + } + /** * Gets the raw offset curve for a line at a given distance. * The quadrant segments, join style and mitre limit can be specified diff --git a/modules/core/src/test/java/org/locationtech/jts/operation/buffer/OffsetCurveTest.java b/modules/core/src/test/java/org/locationtech/jts/operation/buffer/OffsetCurveTest.java index a748a37a3c..9c00b6628c 100644 --- a/modules/core/src/test/java/org/locationtech/jts/operation/buffer/OffsetCurveTest.java +++ b/modules/core/src/test/java/org/locationtech/jts/operation/buffer/OffsetCurveTest.java @@ -382,6 +382,21 @@ public void testMitreJoinNegDistance() { ); } + public void testPolygonMitreJoin() { + checkOffsetCurve( + "POLYGON ((1 1, 1 7, 5 4, 8 8, 8 1, 1 1))", + 1, 0, BufferParameters.JOIN_MITRE, 5, + "LINESTRING (0 0, 0 9, 4.8 5.4, 9 11, 9 0, 0 0)", + 0.0001 + ); + checkOffsetCurve( + "POLYGON ((1 1, 1 7, 5 4, 8 8, 8 1, 1 1))", + -1, 0, BufferParameters.JOIN_MITRE, 5, + "LINESTRING (2 2, 2 5, 5.2 2.6, 7 5, 7 2, 2 2)", + 0.0001 + ); + + } //======================================= private static final double EQUALS_TOL = 0.05;