Mercurial > repos > ecology > xarray_netcdf2netcdf
comparison xarray_mapplot.py @ 0:b0780241f916 draft
"planemo upload for repository https://github.com/galaxyecology/tools-ecology/tree/master/tools/data_manipulation/xarray/ commit 57b6d23e3734d883e71081c78e77964d61be82ba"
| author | ecology | 
|---|---|
| date | Sun, 06 Jun 2021 08:50:11 +0000 | 
| parents | |
| children | e87073edecd6 | 
   comparison
  equal
  deleted
  inserted
  replaced
| -1:000000000000 | 0:b0780241f916 | 
|---|---|
| 1 #!/usr/bin/env python3 | |
| 2 # | |
| 3 # | |
| 4 # usage: xarray_mapplot.py [-h] [--proj PROJ] | |
| 5 # [--cmap CMAP] | |
| 6 # [--output OUTPUT] | |
| 7 # [--time TIMES] | |
| 8 # [--nrow NROW] | |
| 9 # [--ncol NCOL] | |
| 10 # [--title title] | |
| 11 # [--latitude LATITUDE] | |
| 12 # [--longitude LONGITUDE] | |
| 13 # [--land ALPHA-LAND] | |
| 14 # [--ocean ALPHA-OCEAN] | |
| 15 # [--coastline ALPHA-COASTLINE] | |
| 16 # [--borders ALPHA-BORDERS] | |
| 17 # [--xlim "x1,x2"] | |
| 18 # [--ylim "y1,y2"] | |
| 19 # [--range "valmin,valmax"] | |
| 20 # [--threshold VAL] | |
| 21 # [--label label-colorbar] | |
| 22 # [--shift] | |
| 23 # [-v] | |
| 24 # input varname | |
| 25 # | |
| 26 # positional arguments: | |
| 27 # input input filename with geographical coordinates (netCDF | |
| 28 # format) | |
| 29 # varname Specify which variable to plot (case sensitive) | |
| 30 # | |
| 31 # optional arguments: | |
| 32 # -h, --help show this help message and exit | |
| 33 # --proj PROJ Specify the projection on which we draw | |
| 34 # --cmap CMAP Specify which colormap to use for plotting | |
| 35 # --output OUTPUT output filename to store resulting image (png format) | |
| 36 # --time TIMES time index from the file for multiple plots ("0 1 2 3") | |
| 37 # --title plot or subplot title | |
| 38 # --latitude variable name for latitude | |
| 39 # --longitude variable name for longitude | |
| 40 # --land add land on plot with alpha value [0-1] | |
| 41 # --ocean add oceans on plot with alpha value [0-1] | |
| 42 # --coastline add coastline with alpha value [0-1] | |
| 43 # --borders add country borders with alpha value [0-1] | |
| 44 # --xlim limited geographical area longitudes "x1,x2" | |
| 45 # --ylim limited geographical area latitudes "y1,y2" | |
| 46 # --range "valmin,valmax" for plotting | |
| 47 # --threshold do not plot values below threshold | |
| 48 # --label set a label for colormap | |
| 49 # --shift shift longitudes if specified | |
| 50 # -v, --verbose switch on verbose mode | |
| 51 # | |
| 52 | |
| 53 import argparse | |
| 54 import ast | |
| 55 import warnings | |
| 56 from pathlib import Path | |
| 57 | |
| 58 import cartopy.crs as ccrs | |
| 59 import cartopy.feature as feature | |
| 60 | |
| 61 from cmcrameri import cm | |
| 62 | |
| 63 import matplotlib as mpl | |
| 64 mpl.use('Agg') | |
| 65 from matplotlib import pyplot # noqa: I202,E402 | |
| 66 | |
| 67 import xarray as xr # noqa: E402 | |
| 68 | |
| 69 | |
| 70 class MapPlotXr (): | |
| 71 def __init__(self, input, proj, varname, cmap, output, verbose=False, | |
| 72 time=[], title="", latitude="latitude", | |
| 73 longitude="longitude", land=0, ocean=0, | |
| 74 coastline=0, borders=0, xlim=[], ylim=[], | |
| 75 threshold="", label="", shift=False, | |
| 76 range_values=[]): | |
| 77 self.input = input | |
| 78 print("PROJ", proj) | |
| 79 if proj != "" and proj is not None: | |
| 80 self.proj = proj.replace('X', ':') | |
| 81 else: | |
| 82 self.proj = proj | |
| 83 self.varname = varname | |
| 84 self.get_cmap(cmap) | |
| 85 self.time = time | |
| 86 self.latitude = latitude | |
| 87 self.longitude = longitude | |
| 88 self.land = land | |
| 89 self.ocean = ocean | |
| 90 self.coastline = coastline | |
| 91 self.borders = borders | |
| 92 self.xlim = xlim | |
| 93 self.ylim = ylim | |
| 94 self.range = range_values | |
| 95 self.threshold = threshold | |
| 96 self.shift = shift | |
| 97 self.xylim_supported = False | |
| 98 self.colorbar = True | |
| 99 self.title = title | |
| 100 if output is None: | |
| 101 self.output = Path(input).stem + '.png' | |
| 102 else: | |
| 103 self.output = output | |
| 104 self.verbose = verbose | |
| 105 self.dset = xr.open_dataset(self.input, use_cftime=True) | |
| 106 | |
| 107 self.label = {} | |
| 108 if label != "" and label is not None: | |
| 109 self.label['label'] = label | |
| 110 if verbose: | |
| 111 print("input: ", self.input) | |
| 112 print("proj: ", self.proj) | |
| 113 print("varname: ", self.varname) | |
| 114 print("time: ", self.time) | |
| 115 print("minval, maxval: ", self.range) | |
| 116 print("title: ", self.title) | |
| 117 print("output: ", self.output) | |
| 118 print("label: ", self.label) | |
| 119 print("shift: ", self.shift) | |
| 120 print("ocean: ", self.ocean) | |
| 121 print("land: ", self.land) | |
| 122 print("coastline: ", self.coastline) | |
| 123 print("borders: ", self.borders) | |
| 124 print("latitude: ", self.latitude) | |
| 125 print("longitude: ", self.longitude) | |
| 126 print("xlim: ", self.xlim) | |
| 127 print("ylim: ", self.ylim) | |
| 128 | |
| 129 def get_cmap(self, cmap): | |
| 130 if cmap[0:3] == 'cm.': | |
| 131 self.cmap = cm.__dict__[cmap[3:]] | |
| 132 else: | |
| 133 self.cmap = cmap | |
| 134 | |
| 135 def projection(self): | |
| 136 if self.proj is None: | |
| 137 return ccrs.PlateCarree() | |
| 138 | |
| 139 proj_dict = ast.literal_eval(self.proj) | |
| 140 | |
| 141 user_proj = proj_dict.pop("proj") | |
| 142 if user_proj == 'PlateCarree': | |
| 143 self.xylim_supported = True | |
| 144 return ccrs.PlateCarree(**proj_dict) | |
| 145 elif user_proj == 'AlbersEqualArea': | |
| 146 return ccrs.AlbersEqualArea(**proj_dict) | |
| 147 elif user_proj == 'AzimuthalEquidistant': | |
| 148 return ccrs.AzimuthalEquidistant(**proj_dict) | |
| 149 elif user_proj == 'EquidistantConic': | |
| 150 return ccrs.EquidistantConic(**proj_dict) | |
| 151 elif user_proj == 'LambertConformal': | |
| 152 return ccrs.LambertConformal(**proj_dict) | |
| 153 elif user_proj == 'LambertCylindrical': | |
| 154 return ccrs.LambertCylindrical(**proj_dict) | |
| 155 elif user_proj == 'Mercator': | |
| 156 return ccrs.Mercator(**proj_dict) | |
| 157 elif user_proj == 'Miller': | |
| 158 return ccrs.Miller(**proj_dict) | |
| 159 elif user_proj == 'Mollweide': | |
| 160 return ccrs.Mollweide(**proj_dict) | |
| 161 elif user_proj == 'Orthographic': | |
| 162 return ccrs.Orthographic(**proj_dict) | |
| 163 elif user_proj == 'Robinson': | |
| 164 return ccrs.Robinson(**proj_dict) | |
| 165 elif user_proj == 'Sinusoidal': | |
| 166 return ccrs.Sinusoidal(**proj_dict) | |
| 167 elif user_proj == 'Stereographic': | |
| 168 return ccrs.Stereographic(**proj_dict) | |
| 169 elif user_proj == 'TransverseMercator': | |
| 170 return ccrs.TransverseMercator(**proj_dict) | |
| 171 elif user_proj == 'UTM': | |
| 172 return ccrs.UTM(**proj_dict) | |
| 173 elif user_proj == 'InterruptedGoodeHomolosine': | |
| 174 return ccrs.InterruptedGoodeHomolosine(**proj_dict) | |
| 175 elif user_proj == 'RotatedPole': | |
| 176 return ccrs.RotatedPole(**proj_dict) | |
| 177 elif user_proj == 'OSGB': | |
| 178 self.xylim_supported = False | |
| 179 return ccrs.OSGB(**proj_dict) | |
| 180 elif user_proj == 'EuroPP': | |
| 181 self.xylim_supported = False | |
| 182 return ccrs.EuroPP(**proj_dict) | |
| 183 elif user_proj == 'Geostationary': | |
| 184 return ccrs.Geostationary(**proj_dict) | |
| 185 elif user_proj == 'NearsidePerspective': | |
| 186 return ccrs.NearsidePerspective(**proj_dict) | |
| 187 elif user_proj == 'EckertI': | |
| 188 return ccrs.EckertI(**proj_dict) | |
| 189 elif user_proj == 'EckertII': | |
| 190 return ccrs.EckertII(**proj_dict) | |
| 191 elif user_proj == 'EckertIII': | |
| 192 return ccrs.EckertIII(**proj_dict) | |
| 193 elif user_proj == 'EckertIV': | |
| 194 return ccrs.EckertIV(**proj_dict) | |
| 195 elif user_proj == 'EckertV': | |
| 196 return ccrs.EckertV(**proj_dict) | |
| 197 elif user_proj == 'EckertVI': | |
| 198 return ccrs.EckertVI(**proj_dict) | |
| 199 elif user_proj == 'EqualEarth': | |
| 200 return ccrs.EqualEarth(**proj_dict) | |
| 201 elif user_proj == 'Gnomonic': | |
| 202 return ccrs.Gnomonic(**proj_dict) | |
| 203 elif user_proj == 'LambertAzimuthalEqualArea': | |
| 204 return ccrs.LambertAzimuthalEqualArea(**proj_dict) | |
| 205 elif user_proj == 'NorthPolarStereo': | |
| 206 return ccrs.NorthPolarStereo(**proj_dict) | |
| 207 elif user_proj == 'OSNI': | |
| 208 return ccrs.OSNI(**proj_dict) | |
| 209 elif user_proj == 'SouthPolarStereo': | |
| 210 return ccrs.SouthPolarStereo(**proj_dict) | |
| 211 | |
| 212 def plot(self, ts=None): | |
| 213 if self.shift: | |
| 214 if self.longitude == 'longitude': | |
| 215 self.dset = self.dset.assign_coords( | |
| 216 longitude=((( | |
| 217 self.dset[self.longitude] | |
| 218 + 180) % 360) - 180)) | |
| 219 elif self.longitude == 'lon': | |
| 220 self.dset = self.dset.assign_coords( | |
| 221 lon=(((self.dset[self.longitude] | |
| 222 + 180) % 360) - 180)) | |
| 223 | |
| 224 pyplot.figure(1, figsize=[20, 10]) | |
| 225 | |
| 226 # Set the projection to use for plotting | |
| 227 ax = pyplot.subplot(1, 1, 1, projection=self.projection()) | |
| 228 if self.land: | |
| 229 ax.add_feature(feature.LAND, alpha=self.land) | |
| 230 | |
| 231 if self.ocean: | |
| 232 ax.add_feature(feature.OCEAN, alpha=self.ocean) | |
| 233 if self.coastline: | |
| 234 ax.coastlines(resolution='10m', alpha=self.coastline) | |
| 235 if self.borders: | |
| 236 ax.add_feature(feature.BORDERS, linestyle=':', alpha=self.borders) | |
| 237 | |
| 238 if self.xlim: | |
| 239 min_lon = min(self.xlim[0], self.xlim[1]) | |
| 240 max_lon = max(self.xlim[0], self.xlim[1]) | |
| 241 else: | |
| 242 min_lon = self.dset[self.longitude].min() | |
| 243 max_lon = self.dset[self.longitude].max() | |
| 244 | |
| 245 if self.ylim: | |
| 246 min_lat = min(self.ylim[0], self.ylim[1]) | |
| 247 max_lat = max(self.ylim[0], self.ylim[1]) | |
| 248 else: | |
| 249 min_lat = self.dset[self.latitude].min() | |
| 250 max_lat = self.dset[self.latitude].max() | |
| 251 | |
| 252 if self.xylim_supported: | |
| 253 pyplot.xlim(min_lon, max_lon) | |
| 254 pyplot.ylim(min_lat, max_lat) | |
| 255 | |
| 256 # Fix extent | |
| 257 if self.threshold == "" or self.threshold is None: | |
| 258 threshold = self.dset[self.varname].min() | |
| 259 else: | |
| 260 threshold = float(self.threshold) | |
| 261 | |
| 262 if self.range == []: | |
| 263 minval = self.dset[self.varname].min() | |
| 264 maxval = self.dset[self.varname].max() | |
| 265 else: | |
| 266 minval = self.range[0] | |
| 267 maxval = self.range[1] | |
| 268 | |
| 269 if self.verbose: | |
| 270 print("minval: ", minval) | |
| 271 print("maxval: ", maxval) | |
| 272 | |
| 273 # pass extent with vmin and vmax parameters | |
| 274 proj_t = ccrs.PlateCarree() | |
| 275 if ts is None: | |
| 276 self.dset.where( | |
| 277 self.dset[self.varname] > threshold | |
| 278 )[self.varname].plot(ax=ax, | |
| 279 vmin=minval, | |
| 280 vmax=maxval, | |
| 281 transform=proj_t, | |
| 282 cmap=self.cmap, | |
| 283 cbar_kwargs=self.label | |
| 284 ) | |
| 285 if self.title != "" and self.title is not None: | |
| 286 pyplot.title(self.title) | |
| 287 pyplot.savefig(self.output) | |
| 288 else: | |
| 289 if self.colorbar: | |
| 290 self.dset.where( | |
| 291 self.dset[self.varname] > threshold | |
| 292 )[self.varname].isel(time=ts).plot(ax=ax, | |
| 293 vmin=minval, | |
| 294 vmax=maxval, | |
| 295 transform=proj_t, | |
| 296 cmap=self.cmap, | |
| 297 cbar_kwargs=self.label | |
| 298 ) | |
| 299 else: | |
| 300 self.dset.where( | |
| 301 self.dset[self.varname] > minval | |
| 302 )[self.varname].isel(time=ts).plot(ax=ax, | |
| 303 vmin=minval, | |
| 304 vmax=maxval, | |
| 305 transform=proj_t, | |
| 306 cmap=self.cmap, | |
| 307 add_colorbar=False) | |
| 308 if self.title != "" and self.title is not None: | |
| 309 pyplot.title(self.title + "(time = " + str(ts) + ')') | |
| 310 pyplot.savefig(self.output[:-4] + "_time" + str(ts) + | |
| 311 self.output[-4:]) # assume png format | |
| 312 | |
| 313 | |
| 314 if __name__ == '__main__': | |
| 315 warnings.filterwarnings("ignore") | |
| 316 parser = argparse.ArgumentParser() | |
| 317 parser.add_argument( | |
| 318 'input', | |
| 319 help='input filename with geographical coordinates (netCDF format)' | |
| 320 ) | |
| 321 | |
| 322 parser.add_argument( | |
| 323 '--proj', | |
| 324 help='Specify the projection on which we draw' | |
| 325 ) | |
| 326 parser.add_argument( | |
| 327 'varname', | |
| 328 help='Specify which variable to plot (case sensitive)' | |
| 329 ) | |
| 330 parser.add_argument( | |
| 331 '--cmap', | |
| 332 help='Specify which colormap to use for plotting' | |
| 333 ) | |
| 334 parser.add_argument( | |
| 335 '--output', | |
| 336 help='output filename to store resulting image (png format)' | |
| 337 ) | |
| 338 parser.add_argument( | |
| 339 '--time', | |
| 340 help='list of times to plot for multiple plots' | |
| 341 ) | |
| 342 parser.add_argument( | |
| 343 '--title', | |
| 344 help='plot title' | |
| 345 ) | |
| 346 parser.add_argument( | |
| 347 '--latitude', | |
| 348 help='variable name for latitude' | |
| 349 ) | |
| 350 parser.add_argument( | |
| 351 '--longitude', | |
| 352 help='variable name for longitude' | |
| 353 ) | |
| 354 parser.add_argument( | |
| 355 '--land', | |
| 356 help='add land on plot with alpha value [0-1]' | |
| 357 ) | |
| 358 parser.add_argument( | |
| 359 '--ocean', | |
| 360 help='add oceans on plot with alpha value [0-1]' | |
| 361 ) | |
| 362 parser.add_argument( | |
| 363 '--coastline', | |
| 364 help='add coastline with alpha value [0-1]' | |
| 365 ) | |
| 366 parser.add_argument( | |
| 367 '--borders', | |
| 368 help='add country borders with alpha value [0-1]' | |
| 369 ) | |
| 370 parser.add_argument( | |
| 371 '--xlim', | |
| 372 help='limited geographical area longitudes "x1,x2"' | |
| 373 ) | |
| 374 parser.add_argument( | |
| 375 '--ylim', | |
| 376 help='limited geographical area latitudes "y1,y2"' | |
| 377 ) | |
| 378 parser.add_argument( | |
| 379 '--range', | |
| 380 help='min and max values for plotting "minval,maxval"' | |
| 381 ) | |
| 382 parser.add_argument( | |
| 383 '--threshold', | |
| 384 help='do not plot values below threshold' | |
| 385 ) | |
| 386 parser.add_argument( | |
| 387 '--label', | |
| 388 help='set a label for colorbar' | |
| 389 ) | |
| 390 parser.add_argument( | |
| 391 '--shift', | |
| 392 help='shift longitudes if specified', | |
| 393 action="store_true" | |
| 394 ) | |
| 395 parser.add_argument( | |
| 396 "-v", "--verbose", | |
| 397 help="switch on verbose mode", | |
| 398 action="store_true") | |
| 399 args = parser.parse_args() | |
| 400 | |
| 401 if args.time is None: | |
| 402 time = [] | |
| 403 else: | |
| 404 time = list(map(int, args.time.split(","))) | |
| 405 if args.xlim is None: | |
| 406 xlim = [] | |
| 407 else: | |
| 408 xlim = list(map(float, args.xlim.split(","))) | |
| 409 if args.ylim is None: | |
| 410 ylim = [] | |
| 411 else: | |
| 412 ylim = list(map(float, args.ylim.split(","))) | |
| 413 if args.range is None: | |
| 414 range_values = [] | |
| 415 else: | |
| 416 range_values = list(map(float, args.range.split(","))) | |
| 417 if args.latitude is None: | |
| 418 latitude = "latitude" | |
| 419 else: | |
| 420 latitude = args.latitude | |
| 421 if args.longitude is None: | |
| 422 longitude = "longitude" | |
| 423 else: | |
| 424 longitude = args.longitude | |
| 425 if args.land is None: | |
| 426 land = 0 | |
| 427 else: | |
| 428 land = float(args.land) | |
| 429 if args.ocean is None: | |
| 430 ocean = 0 | |
| 431 else: | |
| 432 ocean = float(args.ocean) | |
| 433 if args.coastline is None: | |
| 434 coastline = 0 | |
| 435 else: | |
| 436 coastline = float(args.coastline) | |
| 437 if args.borders is None: | |
| 438 borders = 0 | |
| 439 else: | |
| 440 borders = float(args.borders) | |
| 441 | |
| 442 dset = MapPlotXr(input=args.input, proj=args.proj, varname=args.varname, | |
| 443 cmap=args.cmap, output=args.output, verbose=args.verbose, | |
| 444 time=time, title=args.title, | |
| 445 latitude=latitude, longitude=longitude, land=land, | |
| 446 ocean=ocean, coastline=coastline, borders=borders, | |
| 447 xlim=xlim, ylim=ylim, threshold=args.threshold, | |
| 448 label=args.label, shift=args.shift, | |
| 449 range_values=range_values) | |
| 450 | |
| 451 if dset.time == []: | |
| 452 dset.plot() | |
| 453 else: | |
| 454 for t in dset.time: | |
| 455 dset.plot(t) | |
| 456 dset.shift = False # only shift once | |
| 457 dset.colorbar = True | 
