From 34043ab58eb254feee0c2f47cb63e057905b49d7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Dav=C3=ADd=20Brakenhoff?= Date: Sat, 26 Oct 2024 00:24:41 +0200 Subject: [PATCH] fix(gridintersect): fix multiple issues (#2343) Fix #2342: * fix typo in shapely.multilinestrings * fix issue with np.apply_along_axis * consider geometry collections when computing overlaps * add test for vertex mode * add inactive test for structured mode (not yet working) Notes: * np.apply_along_axis was reducing result from multiple GeometryCollections to a single MultiLineString causing duplication of shapes in the intersection result. * structured support will be dropped in future releases (see #2344) --- autotest/test_gridintersect.py | 20 ++++++++++++++++++++ flopy/utils/gridintersect.py | 22 ++++++++++++++-------- 2 files changed, 34 insertions(+), 8 deletions(-) diff --git a/autotest/test_gridintersect.py b/autotest/test_gridintersect.py index 83d7672d3..0fccfb4e5 100644 --- a/autotest/test_gridintersect.py +++ b/autotest/test_gridintersect.py @@ -775,6 +775,26 @@ def test_tri_grid_linestring_cell_boundary_return_all_ix_shapely(rtree): assert len(r) == 3 +@requires_pkg("shapely") +def test_rect_vertex_grid_linestring_geomcollection(): + gr = get_rect_vertex_grid() + ix = GridIntersect(gr, method="vertex") + ls = LineString( + [ + (20.0, 0.0), + (5.0, 5.0), + (15.0, 7.5), + (10.0, 10.0), + (5.0, 15.0), + (10.0, 19.0), + (10.0, 20.0), + ] + ) + result = ix.intersect(ls) + assert len(result) == 3 + assert np.allclose(result.lengths.sum(), ls.length) + + # %% test polygon structured diff --git a/flopy/utils/gridintersect.py b/flopy/utils/gridintersect.py index 52bdcd155..e31631f6d 100644 --- a/flopy/utils/gridintersect.py +++ b/flopy/utils/gridintersect.py @@ -547,7 +547,7 @@ def parse_linestrings_in_geom_collection(gc): parts = shapely.get_parts(gc) parts = parts[np.isin(shapely.get_type_id(parts), line_ids)] if len(parts) > 1: - p = shapely.multilinestring(parts) + p = shapely.multilinestrings(parts) elif len(parts) == 0: p = shapely.LineString() else: @@ -558,11 +558,17 @@ def parse_linestrings_in_geom_collection(gc): geomtype_ids[~mask_empty & mask_type] == shapely.GeometryType.GEOMETRYCOLLECTION ) - ixresult[mask_gc] = np.apply_along_axis( - parse_linestrings_in_geom_collection, - axis=0, - arr=ixresult[mask_gc], - ) + # NOTE: not working for multiple geometry collections, result is reduced + # to a single multilinestring, which causes doubles in the result + # ixresult[mask_gc] = np.apply_along_axis( + # parse_linestrings_in_geom_collection, + # axis=0, + # arr=ixresult[mask_gc], + # ) + ixresult[mask_gc] = [ + parse_linestrings_in_geom_collection(gc) + for gc in ixresult[mask_gc] + ] if not return_all_intersections: # intersection with grid cell boundaries @@ -570,7 +576,7 @@ def parse_linestrings_in_geom_collection(gc): shp, shapely.get_exterior_ring(self.geoms[qcellids]) ) mask_bnds_empty = shapely.is_empty(ixbounds) - mask_bnds_type = np.isin(shapely.get_type_id(ixbounds), line_ids) + mask_bnds_type = np.isin(shapely.get_type_id(ixbounds), all_ids) # get ids of boundary intersections idxs = np.nonzero(~mask_bnds_empty & mask_bnds_type)[0] @@ -586,7 +592,7 @@ def parse_linestrings_in_geom_collection(gc): mask_bnds_empty = shapely.is_empty( isect ) # select boundary ix result - mask_overlap = np.isin(shapely.get_type_id(isect), line_ids) + mask_overlap = np.isin(shapely.get_type_id(isect), all_ids) # calculate difference between self and overlapping result diff = shapely.difference(