Cartopy with gridded data#
This example demonstrates:
plotting gridded data with cartopy
reprojecting a map
setting a nodata value to render as a different color
import xarray as xr
import matplotlib.pyplot as plt
import cartopy.crs as ccrs
---------------------------------------------------------------------------
ModuleNotFoundError Traceback (most recent call last)
Cell In[1], line 1
----> 1 import xarray as xr
2 import matplotlib.pyplot as plt
3 import cartopy.crs as ccrs
ModuleNotFoundError: No module named 'xarray'
sst_ds = xr.open_dataset('../lessons/gridded_data/data/oisst-avhrr-v02r01.20220304.nc')
sst = sst_ds.sst.isel(zlev=0)
# Preprocessing - convert longitude to -180 -> 180 degrees
sst['lon'] = (sst['lon'] + 180) % 360 - 180
sst = sst.sortby(sst.lon)
Quick plots with xarray#
# Selecting one time step to plot
sst.isel(time=0).plot()
<matplotlib.collections.QuadMesh at 0x168304640>
data:image/s3,"s3://crabby-images/6ebf6/6ebf6b5afea99e67d84d8d75ee2c91f9f4eb4700" alt="../_images/debc5135b80f248cb63922ac2b01e298557397fcecc66a286298ddd7be1f3e40.png"
Set the nodata value to grey#
“bwr” is the name of the colormap that I chose for this image. You can change “bwr” to any other colormap you want. Ex. plt.cm.get_cmap("viridis").copy()
Grey can also be changed to any color. I like using grey
because it is clear, but I often prefer a slightly lighter shade than the default, so I’ll use the hexadecimal code #c0c0c0
. (Example below in a comment)
# Extract the colormap that I want "bwr"
bwr_badgrey = plt.cm.get_cmap("bwr").copy()
bwr_badgrey.set_bad('grey')
# bwr_badgrey.set_bad('#c0c0c0')
sst.isel(time=0).plot(cmap=bwr_badgrey)
<matplotlib.collections.QuadMesh at 0x168480be0>
data:image/s3,"s3://crabby-images/86171/8617157c996ef5a2501fd1799079dab7e13f8702" alt="../_images/dab19cbf77d67dc60414e79afbeaee33a8d3b84339ebaefef68ea101c698e45c.png"
More detailed plots with cartopy#
# Set the projection. PlateCarree is "regular" lat/lon
map_proj = ccrs.PlateCarree()
data_proj = ccrs.PlateCarree()
sst_AK = sst.sel(lon=slice(-180, -120), lat=slice(40, 75)).isel(time=0)
sst_AK.plot()
<matplotlib.collections.QuadMesh at 0x168545cf0>
data:image/s3,"s3://crabby-images/532af/532af687e3587861b99e467151e1140a9ba8fa6c" alt="../_images/2fa67f548c4352ef023e7abf3dba7a0f97c7c7785042620450ac00fda3f2f88c.png"
fig = plt.figure()
ax = plt.axes(projection=map_proj)
# Add the data to the plot
pc = ax.pcolormesh(sst_AK.lon, sst_AK.lat, sst_AK, transform=data_proj)
# Add gridlines
gl = ax.gridlines(draw_labels=True, linewidth=0.5, linestyle='--', alpha=0.9)
gl.top_labels, gl.right_labels = False, False
# the previous line to removes top and right gridlines labels, which overlap
# the colorbar
# Add the colorbar
cb = fig.colorbar(pc, extend='neither')
cb.set_label('SST [Celsius]', rotation=270, labelpad=18)
# Add a title
fig.suptitle('SST March 3 2022')
# Add coastlines
ax.coastlines()
<cartopy.mpl.feature_artist.FeatureArtist at 0x1686d5240>
data:image/s3,"s3://crabby-images/d2adc/d2adc962cdb38388d5e3411e36539411264f24a2" alt="../_images/da2ab7f4543d96e039dade074e682c4ae87cb83d977fcd007d466c4dcb838791.png"
Transforming the map to another projection#
Alaska is pretty far north, so we may want to use a map projection that is more accurate for that region. Looking through the cartopy projection list it looks like an Albers Equal Area projection minimizes distortion in northern latitudes, so let’s use that one.
# Define a new map projection
map_proj = ccrs.AlbersEqualArea(central_longitude=-150)
Copy and paste the exact same code as above
fig = plt.figure()
ax = plt.axes(projection=map_proj)
# Add the data to the plot
pc = ax.pcolormesh(sst_AK.lon, sst_AK.lat, sst_AK, transform=data_proj)
# Add gridlines
gl = ax.gridlines(draw_labels=True, linewidth=0.5, linestyle='--', alpha=0.9)
gl.top_labels, gl.right_labels = False, False
# the previous line to removes top and right gridlines labels, which overlap
# the colorbar
# Add the colorbar
cb = fig.colorbar(pc, extend='neither')
cb.set_label('SST [Celsius]', rotation=270, labelpad=18)
# Add a title
fig.suptitle('SST March 3 2022')
# Add coastlines
ax.coastlines()
<cartopy.mpl.feature_artist.FeatureArtist at 0x168ca5f30>
data:image/s3,"s3://crabby-images/2ee6e/2ee6e235391258df8508a1acb3869cb5e2a6c79c" alt="../_images/a284a935f4897d0aad5e6b5ae71cc47bb80da5c547b104a802932fff492618f1.png"
Much less distorted!