Source code for spec2nexus.charts

#!/usr/bin/env python

"""
charting for spec2nexus

.. autosummary::

    ~make_png
    ~xy_plot

"""

import datetime
import numpy
import matplotlib
import matplotlib.figure

matplotlib.use("Agg")
from matplotlib.backends.backend_agg import FigureCanvasAgg as FigureCanvas

import spec2nexus


SCALING_FACTOR = 1  #  2**24
PLOT_H_INT = 9  # 7
PLOT_V_INT = 5  # 3
COLORMAP = "cubehelix"  # http://matplotlib.org/api/pyplot_summary.html#matplotlib.pyplot.colormaps
WATERMARK_TEXT = "%s, (C) %s" % (
    spec2nexus.__package_name__,
    spec2nexus.__copyright__.split(",")[0],
)


[docs]def make_png( image, image_file, axes=None, title="2-D data", subtitle="", log_image=False, hsize=PLOT_H_INT, vsize=PLOT_V_INT, cmap=COLORMAP, xtitle=None, ytitle=None, timestamp_str=None, ): """ read the image from the named HDF5 file and make a PNG file Test that the HDF5 file exists and that the path to the data exists in that file. Read the data from the named dataset, mask off some bad values, convert to log(image) and use Matplotlib to make the PNG file. :param obj image: array of data to be rendered :param str image_file: name of image file to be written (path is optional) :param bool log_image: plot log(image) :param int hsize: horizontal size of the PNG image (default: 7) :param int hsize: vertical size of the PNG image (default: 3) :param str cmap: colormap for the image (default: 'cubehelix'), 'jet' is another good one :return str: *image_file* The HDF5 file could be a NeXus file, or some other layout. """ # replace masked data with min good value image_data = numpy.ma.masked_less_equal(image, 0) image_data = image_data.filled(image_data.min()) image_data = numpy.ma.masked_invalid( image_data ) # mask out any NaN values if log_image and image_data.max() != 0: # apply log scaling image_data = numpy.log(image_data) image_data -= image_data.min() image_data *= SCALING_FACTOR / image_data.max() fig = matplotlib.figure.Figure(figsize=(hsize, vsize)) fig.clf() ax = fig.add_subplot(1, 1, 1) if isinstance(axes, list) and len(axes) == 2: def shift_to_pixel_boundaries(arr): np_arr = numpy.array(arr) step = np_arr[1] - np_arr[0] return numpy.append(np_arr, np_arr[-1] + step) - step / 2 x = shift_to_pixel_boundaries(axes[0]) y = shift_to_pixel_boundaries(axes[1]) ax.pcolor(x, y, image_data, cmap=cmap) ax.axis([x.min(), x.max(), y.min(), y.max()]) else: ax.imshow(image_data, interpolation="nearest", cmap=cmap) if xtitle is not None: ax.set_xlabel(xtitle) if ytitle is not None: ax.set_ylabel(ytitle) timestamp_str = timestamp_str or str(datetime.datetime.now()) if subtitle is not None: ax.set_title(subtitle, fontsize=10) fig.suptitle(title, fontsize=8) fig.text( 0.02, 0.0, timestamp_str, fontsize=8, color="gray", ha="left", va="bottom", alpha=0.5, ) fig.text( 0.98, 0.0, WATERMARK_TEXT, fontsize=8, color="gray", ha="right", va="bottom", alpha=0.5, ) ax.ticklabel_format(useOffset=False, style="plain") FigureCanvas(fig).print_figure(image_file, bbox_inches="tight") return image_file
[docs]def xy_plot( x, y, plot_file, title=None, subtitle=None, xtitle=None, ytitle=None, xlog=False, ylog=False, hsize=PLOT_H_INT, vsize=PLOT_V_INT, timestamp_str=None, ): """ with MatPlotLib, generate a plot of a scan (as if data from a scan in a SPEC file) :param [float] x: horizontal axis data :param [float] y: vertical axis data :param str plot_file: file name to write plot image :param str xtitle: horizontal axis label (default: not shown) :param str ytitle: vertical axis label (default: not shown) :param str title: title for plot (default: date time) :param str subtitle: subtitle for plot (default: not shown) :param bool xlog: should X axis be log (default: False=linear) :param bool ylog: should Y axis be log (default: False=linear) :param str timestamp_str: date to use on plot (default: now) .. tip:: when using this module as a background task ... MatPlotLib has several interfaces for plotting. Since this module runs as part of a background job generating lots of plots, MatPlotLib's standard ``plt`` code is not the right model. It warns after 20 plots and will eventually run out of memory. Here's the fix used in this module: http://stackoverflow.com/questions/16334588/create-a-figure-that-is-reference-counted/16337909#16337909 """ fig = matplotlib.figure.Figure(figsize=(hsize, vsize)) fig.clf() ax = fig.add_subplot(1, 1, 1) if xlog: ax.set_xscale("log") if max(x) <= 0: msg = "X data has no positive values," msg += " and therefore can not be log-scaled." raise ValueError(msg) if ylog: ax.set_yscale("log") if max(y) <= 0: msg = "Y data has no positive values," msg += " and therefore can not be log-scaled." raise ValueError(msg) if not xlog and not ylog: ax.ticklabel_format(useOffset=False) if xtitle is not None: ax.set_xlabel(xtitle) if ytitle is not None: ax.set_ylabel(ytitle) if subtitle is not None: ax.set_title(subtitle, fontsize=10) fig.suptitle(title, fontsize=8) fig.text( 0.02, 0.0, timestamp_str, fontsize=8, color="gray", ha="left", va="bottom", alpha=0.5, ) fig.text( 0.98, 0.0, WATERMARK_TEXT, fontsize=8, color="gray", ha="right", va="bottom", alpha=0.5, ) if not xlog and not ylog: ax.ticklabel_format(useOffset=False, style="plain") ax.plot(x, y, "o-") FigureCanvas(fig).print_figure(plot_file, bbox_inches="tight")