Skip to content

Commit

Permalink
Merge branch 'SciTools:main' into wmtscache
Browse files Browse the repository at this point in the history
  • Loading branch information
dnowacki-usgs authored Dec 30, 2024
2 parents c98d21c + 113be8e commit 326a78c
Show file tree
Hide file tree
Showing 7 changed files with 63 additions and 33 deletions.
4 changes: 2 additions & 2 deletions .github/workflows/release.yml
Original file line number Diff line number Diff line change
Expand Up @@ -98,7 +98,7 @@ jobs:
path: dist

- name: Build wheels for CPython
uses: pypa/cibuildwheel@7940a4c0e76eb2030e473a5f864f291f63ee879b # v2.21.3
uses: pypa/cibuildwheel@ee63bf16da6cddfb925f542f2c7b59ad50e93969 # v2.22.0
with:
package-dir: dist/${{ needs.build_sdist.outputs.SDIST_NAME }}
env:
Expand Down Expand Up @@ -134,4 +134,4 @@ jobs:
merge-multiple: true

- name: Publish Package
uses: pypa/[email protected].2
uses: pypa/[email protected].3
2 changes: 1 addition & 1 deletion docs/source/citation.rst
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ For example::

@manual{Cartopy,
author = {{Met Office}},
title = {Cartopy: a cartographic python library with a Matplotlib interface},
title = {Cartopy: a cartographic {Python} library with a {Matplotlib} interface},
year = {2010 - 2015},
address = {Exeter, Devon },
url = {https://scitools.org.uk/cartopy}
Expand Down
53 changes: 37 additions & 16 deletions lib/cartopy/crs.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
import pyproj
from pyproj import Transformer
from pyproj.exceptions import ProjError
import shapely
import shapely.geometry as sgeom
from shapely.prepared import prep

Expand Down Expand Up @@ -1214,34 +1215,54 @@ def _rings_to_multi_polygon(self, rings, is_ccw):
y3 -= by
x4 += bx
y4 += by

interior_polys = []

for ring in interior_rings:
# Use shapely buffer in an attempt to fix invalid geometries
polygon = sgeom.Polygon(ring).buffer(0)
if not polygon.is_empty and polygon.is_valid:
polygon = shapely.make_valid(sgeom.Polygon(ring))
if not polygon.is_empty:
if isinstance(polygon, sgeom.Polygon):
interior_polys.append(polygon)
elif isinstance(polygon, sgeom.MultiPolygon):
interior_polys.extend(polygon.geoms)
elif isinstance(polygon, sgeom.GeometryCollection):
for geom in polygon.geoms:
if isinstance(geom, sgeom.Polygon):
interior_polys.append(geom)
elif isinstance(geom, sgeom.MultiPolygon):
interior_polys.extend(geom.geoms)
else:
# make_valid may produce some linestrings. Ignore these
continue

x1, y1, x2, y2 = polygon.bounds
bx = (x2 - x1) * 0.1
by = (y2 - y1) * 0.1
x1 -= bx
y1 -= by
x2 += bx
y2 += by
box = sgeom.box(min(x1, x3), min(y1, y3),
max(x2, x4), max(y2, y4))

# Invert the polygon
polygon = box.difference(polygon)
x3 = min(x1, x3)
x4 = max(x2, x4)
y3 = min(y1, y3)
y4 = max(y2, y4)

# Intersect the inverted polygon with the boundary
polygon = boundary_poly.intersection(polygon)
box = sgeom.box(x3, y3, x4, y4, ccw=is_ccw)

if not polygon.is_empty:
polygon_bits.append(polygon)
# Invert the polygons
polygon = box.difference(sgeom.MultiPolygon(interior_polys))

if polygon_bits:
multi_poly = sgeom.MultiPolygon(polygon_bits)
else:
multi_poly = sgeom.MultiPolygon()
return multi_poly
# Intersect the inverted polygon with the boundary
polygon = boundary_poly.intersection(polygon)

if not polygon.is_empty:
if isinstance(polygon, sgeom.MultiPolygon):
polygon_bits.extend(polygon.geoms)
else:
polygon_bits.append(polygon)

return sgeom.MultiPolygon(polygon_bits)

def quick_vertices_transform(self, vertices, src_crs):
"""
Expand Down
22 changes: 11 additions & 11 deletions lib/cartopy/mpl/gridliner.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
import matplotlib.ticker as mticker
import matplotlib.transforms as mtrans
import numpy as np
import shapely
import shapely.geometry as sgeom

import cartopy
Expand Down Expand Up @@ -815,17 +816,16 @@ def update_artist(artist, renderer):
if isinstance(intersection, sgeom.LineString):
intersection = [intersection]
elif len(intersection.geoms) > 4:
# Gridline and map boundary are parallel and they
# intersect themselves too much it results in a
# multiline string that must be converted to a single
# linestring. This is an empirical workaround for a
# problem that can probably be solved in a cleaner way.
xy = np.append(
intersection.geoms[0].coords,
intersection.geoms[-1].coords,
axis=0,
)
intersection = [sgeom.LineString(xy)]
# If lines are parallel, there will be many intersections
# merge them to get only one for the calculations below
merged_line = shapely.line_merge(intersection)
if isinstance(merged_line, sgeom.MultiLineString):
# our merge still produced a multilinestring, so
# manually concatenate the original coordinates
xy = np.concatenate(
[inter.coords for inter in intersection.geoms], axis=0)
merged_line = shapely.LineString(xy)
intersection = [merged_line]
else:
intersection = intersection.geoms
tails = []
Expand Down
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
3 changes: 0 additions & 3 deletions lib/cartopy/tests/mpl/test_gridliner.py
Original file line number Diff line number Diff line change
Expand Up @@ -149,13 +149,10 @@ def test_grid_labels():
ax.coastlines(resolution="110m")
ax.gridlines(draw_labels=True)

# Check that adding labels to Mercator gridlines gives an error.
# (Currently can only label PlateCarree gridlines.)
ax = fig.add_subplot(
3, 2, 2, projection=ccrs.PlateCarree(central_longitude=180))
ax.coastlines(resolution="110m")

ax.set_title('Known bug')
gl = ax.gridlines(crs=crs_pc, draw_labels=True)
gl.top_labels = False
gl.left_labels = False
Expand Down
12 changes: 12 additions & 0 deletions lib/cartopy/tests/test_polygon.py
Original file line number Diff line number Diff line change
Expand Up @@ -446,6 +446,18 @@ def test_inverted_poly_simple_hole(self):
self._assert_bounds(polygon.interiors[0].bounds,
- 1.2e7, -1.2e7, 1.2e7, 1.2e7, 1e6)

def test_inverted_poly_multi_hole(self):
# Adapted from 1149
proj = ccrs.LambertAzimuthalEqualArea(
central_latitude=45, central_longitude=-100)
poly = sgeom.Polygon([(-180, -80), (180, -80), (180, 90), (-180, 90)],
[[(-50, -50), (-50, 0), (0, 0), (0, -50)]])
multi_polygon = proj.project_geometry(poly)
# Should project to single polygon with multiple holes
assert len(multi_polygon.geoms) == 1
assert len(multi_polygon.geoms[0].interiors) >= 2


def test_inverted_poly_clipped_hole(self):
proj = ccrs.NorthPolarStereo()
poly = sgeom.Polygon([(0, 0), (-90, 0), (-180, 0), (-270, 0)],
Expand Down

0 comments on commit 326a78c

Please sign in to comment.