diff --git a/tools/psy-maps/psy-maps.xml b/tools/psy-maps/psy-maps.xml index 631a49b..da9fdba 100644 --- a/tools/psy-maps/psy-maps.xml +++ b/tools/psy-maps/psy-maps.xml @@ -1,4 +1,4 @@ - + gridded (lat/lon) netCDF data topic_3855 @@ -9,15 +9,15 @@ python - psyplot - psy-maps - psy-reg - netcdf4 + psyplot + psy-maps + psy-reg + netcdf4 + + + + @@ -162,6 +166,14 @@ + + + + + + + + + 10.21105/joss.00363 diff --git a/tools/psy-maps/psymap_simple.py b/tools/psy-maps/psymap_simple.py index 00b7cad..e860ef5 100644 --- a/tools/psy-maps/psymap_simple.py +++ b/tools/psy-maps/psymap_simple.py @@ -4,6 +4,7 @@ # usage: psymap_simple.py [-h] [--proj PROJ] # [--cmap CMAP] # [--output OUTPUT] +# [-l] # [-v] # input varname # @@ -14,6 +15,7 @@ # # optional arguments: # -h, --help show this help message and exit +# -l, --logscale log scale the data # --proj PROJ Specify the projection on which we draw # --cmap CMAP Specify which colormap to use for plotting # --output OUTPUT output filename to store resulting image (png format) @@ -26,47 +28,63 @@ # import argparse +import math import warnings from pathlib import Path import matplotlib as mpl +import psyplot.project as psy # noqa: I202,E402 +import xarray + mpl.use('Agg') from matplotlib import pyplot # noqa: I202,E402 - -import psyplot.project as psy # noqa: I202,E402 from psyplot import rcParams # noqa: I202,E402 class PsyPlot (): - def __init__(self, input, proj, varname, cmap, output, verbose=False, - time=[], nrow=1, ncol=1, format="%B %e, %Y", - title=""): + def __init__(self, input, varname, output=None, logscale=False, cmap=None, + proj=None, verbose=False, time=None, nrow=None, ncol=None, + format=None, title=None): self.input = input - self.proj = proj self.varname = varname - self.cmap = cmap - self.time = time - if format is None: - self.format = "" - else: - self.format = format.replace('X', '%') - if title is None: - self.title = "" + if proj is None or proj == "": + self.proj = "cyl" else: + self.proj = proj + self.cmap = cmap if cmap is not None else "jet" + self.time = time if time is not None else [] + self.ncol = int(ncol) if ncol is not None else int(1) + self.nrow = int(nrow) if nrow is not None else int(1) + + # Open dataset + ds = xarray.open_dataset(input)[varname] + minv = math.log2(ds.data.min()) + maxv = math.log2(ds.data.max()) + if title is not None: self.title = title - if ncol is None: - self.ncol = 1 else: - self.ncol = int(ncol) - if nrow is None: - self.nrow = 1 + self.title = ds.long_name + if len(self.title) > 60: + self.title = ds.standard_name + del ds + + if logscale: + # Check that data range is sufficient for log scaling + if maxv < (minv * (10.0 ** 2.0)): + print("Not possible to log scale, switching to linear scale") + self.bounds = None + else: + self.bounds = ['log', 2] + else: + self.bounds = None + if format is None: + self.format = "" else: - self.nrow = int(nrow) + self.format = "%B %e, %Y" if output is None: self.output = Path(input).stem + '.png' else: self.output = output - self.verbose = verbose if verbose: print("input: ", self.input) print("proj: ", self.proj) @@ -77,37 +95,32 @@ def __init__(self, input, proj, varname, cmap, output, verbose=False, print("nrow: ", self.nrow) print("title: ", self.title) print("date format: ", self.format) + print("logscale: ", self.bounds) print("output: ", self.output) def plot(self): + clabel = '{desc}' if self.title and self.format: title = self.title + "\n" + self.format elif not self.title and self.format: title = self.format elif self.title and not self.format: title = self.title - else: - title = '%(long_name)s' + clabel = self.title - if self.cmap is None and self.proj is None: - psy.plot.mapplot(self.input, name=self.varname, - title=title, - clabel='{desc}') - elif self.proj is None or not self.proj: - psy.plot.mapplot(self.input, name=self.varname, - title=title, - cmap=self.cmap, clabel='{desc}') - elif self.cmap is None or not self.cmap: + # Plot with chosen options + if self.bounds is None: psy.plot.mapplot(self.input, name=self.varname, + cmap=self.cmap, projection=self.proj, title=title, - clabel='{desc}') + clabel=clabel) else: psy.plot.mapplot(self.input, name=self.varname, - cmap=self.cmap, + cmap=self.cmap, bounds=self.bounds, projection=self.proj, title=title, - clabel='{desc}') + clabel=clabel) pyplot.savefig(self.output) @@ -119,50 +132,41 @@ def multiple_plot(self): title = self.format else: title = self.title + "\n" + self.format + mpl.rcParams['figure.figsize'] = [20, 8] mpl.rcParams.update({'font.size': 8}) rcParams.update({'plotter.maps.grid_labelsize': 8.0}) - if self.cmap is None and self.proj is None: - m = psy.plot.mapplot(self.input, name=self.varname, - title=title, - ax=(self.nrow, self.ncol), - time=self.time, sort=['time'], - clabel='{desc}') - m.share(keys='bounds') - elif self.proj is None or not self.proj: - m = psy.plot.mapplot(self.input, name=self.varname, - title=title, - ax=(self.nrow, self.ncol), - time=self.time, sort=['time'], - cmap=self.cmap, clabel='{desc}') - m.share(keys='bounds') - elif self.cmap is None or not self.cmap: + + # Plot using options + if self.bounds is None: m = psy.plot.mapplot(self.input, name=self.varname, + cmap=self.cmap, projection=self.proj, ax=(self.nrow, self.ncol), time=self.time, sort=['time'], title=title, clabel='{desc}') - m.share(keys='bounds') else: m = psy.plot.mapplot(self.input, name=self.varname, - cmap=self.cmap, + cmap=self.cmap, bounds=self.bounds, projection=self.proj, ax=(self.nrow, self.ncol), time=self.time, sort=['time'], title=title, clabel='{desc}') - m.share(keys='bounds') + + m.share(keys='bounds') pyplot.savefig(self.output) -def psymap_plot(input, proj, varname, cmap, output, verbose, time, +def psymap_plot(input, proj, varname, logscale, cmap, output, verbose, time, nrow, ncol, format, title): """Generate plot from input filename""" - p = PsyPlot(input, proj, varname, cmap, output, verbose, time, + p = PsyPlot(input, varname, output, logscale, cmap, proj, verbose, time, nrow, ncol, format, title) + if len(time) == 0: p.plot() else: @@ -185,6 +189,10 @@ def psymap_plot(input, proj, varname, cmap, output, verbose, time, 'varname', help='Specify which variable to plot (case sensitive)' ) + parser.add_argument( + "--logscale", + help='Plot the log scaled data' + ) parser.add_argument( '--cmap', help='Specify which colormap to use for plotting' @@ -223,6 +231,12 @@ def psymap_plot(input, proj, varname, cmap, output, verbose, time, time = [] else: time = list(map(int, args.time.split(","))) - psymap_plot(args.input, args.proj, args.varname, args.cmap, + + if args.logscale == 'no': + logscale = False + else: + logscale = True + + psymap_plot(args.input, args.proj, args.varname, logscale, args.cmap, args.output, args.verbose, time, args.nrow, args.ncol, args.format, args.title) diff --git a/tools/psy-maps/test-data/ESACCI-OC-L3S-CHLOR_A-20220601-RESIZED-fv6.0.nc b/tools/psy-maps/test-data/ESACCI-OC-L3S-CHLOR_A-20220601-RESIZED-fv6.0.nc new file mode 100644 index 0000000..0439bd4 Binary files /dev/null and b/tools/psy-maps/test-data/ESACCI-OC-L3S-CHLOR_A-20220601-RESIZED-fv6.0.nc differ diff --git a/tools/psy-maps/test-data/ESACCI-OC-L3S-CHLOR_A-20220601-RESIZED-fv6.0.png b/tools/psy-maps/test-data/ESACCI-OC-L3S-CHLOR_A-20220601-RESIZED-fv6.0.png new file mode 100644 index 0000000..0188f47 Binary files /dev/null and b/tools/psy-maps/test-data/ESACCI-OC-L3S-CHLOR_A-20220601-RESIZED-fv6.0.png differ diff --git a/tools/psy-maps/test-data/TS.f2000.T31T31.control.cam.h0.0014-12.png b/tools/psy-maps/test-data/TS.f2000.T31T31.control.cam.h0.0014-12.png index 4e4d3f9..9b411ab 100644 Binary files a/tools/psy-maps/test-data/TS.f2000.T31T31.control.cam.h0.0014-12.png and b/tools/psy-maps/test-data/TS.f2000.T31T31.control.cam.h0.0014-12.png differ diff --git a/tools/psy-maps/test-data/TS.f2000.T31T31.control.cam.h0.0014-12_ortho.png b/tools/psy-maps/test-data/TS.f2000.T31T31.control.cam.h0.0014-12_ortho.png index 2938317..60afcd0 100644 Binary files a/tools/psy-maps/test-data/TS.f2000.T31T31.control.cam.h0.0014-12_ortho.png and b/tools/psy-maps/test-data/TS.f2000.T31T31.control.cam.h0.0014-12_ortho.png differ