backend_inline.py
227 lines
| 8.5 KiB
| text/x-python
|
PythonLexer
Fernando Perez
|
r2890 | """Produce SVG versions of active plots for display by the rich Qt frontend. | ||
""" | ||||
#----------------------------------------------------------------------------- | ||||
# Imports | ||||
#----------------------------------------------------------------------------- | ||||
Fernando Perez
|
r2987 | from __future__ import print_function | ||
Fernando Perez
|
r2890 | |||
epatters
|
r2756 | # Standard library imports | ||
Fernando Perez
|
r3731 | import sys | ||
epatters
|
r2756 | |||
Fernando Perez
|
r3731 | # Third-party imports | ||
Fernando Perez
|
r2987 | import matplotlib | ||
MinRK
|
r3973 | from matplotlib.backends.backend_agg import new_figure_manager | ||
epatters
|
r2756 | from matplotlib._pylab_helpers import Gcf | ||
# Local imports. | ||||
MinRK
|
r3973 | from IPython.config.configurable import SingletonConfigurable | ||
Brian Granger
|
r3277 | from IPython.core.displaypub import publish_display_data | ||
Fernando Perez
|
r5469 | from IPython.core.pylabtools import print_figure, select_figure_format | ||
MinRK
|
r5087 | from IPython.utils.traitlets import Dict, Instance, CaselessStrEnum, CBool | ||
MinRK
|
r5227 | from IPython.utils.warn import warn | ||
MinRK
|
r3973 | #----------------------------------------------------------------------------- | ||
# Configurable for inline backend options | ||||
#----------------------------------------------------------------------------- | ||||
MinRK
|
r5227 | # inherit from InlineBackendConfig for deprecation purposes | ||
MinRK
|
r3973 | class InlineBackendConfig(SingletonConfigurable): | ||
MinRK
|
r5227 | pass | ||
class InlineBackend(InlineBackendConfig): | ||||
MinRK
|
r3973 | """An object to store configuration of the inline backend.""" | ||
MinRK
|
r5227 | def _config_changed(self, name, old, new): | ||
# warn on change of renamed config section | ||||
if new.InlineBackendConfig != old.InlineBackendConfig: | ||||
warn("InlineBackendConfig has been renamed to InlineBackend") | ||||
super(InlineBackend, self)._config_changed(name, old, new) | ||||
MinRK
|
r3973 | # The typical default figure size is too large for inline use, | ||
# so we shrink the figure size to 6x4, and tweak fonts to | ||||
MinRK
|
r5292 | # make that fit. | ||
MinRK
|
r3973 | rc = Dict({'figure.figsize': (6.0,4.0), | ||
Takafumi Arakaki
|
r7798 | # play nicely with white background in the Qt and notebook frontend | ||
'figure.facecolor': 'white', | ||||
'figure.edgecolor': 'white', | ||||
MinRK
|
r3973 | # 12pt labels get cutoff on 6x4 logplots, so use 10pt. | ||
'font.size': 10, | ||||
MinRK
|
r5292 | # 72 dpi matches SVG/qtconsole | ||
# this only affects PNG export, as SVG has no dpi setting | ||||
'savefig.dpi': 72, | ||||
MinRK
|
r3973 | # 10pt still needs a little more room on the xlabel: | ||
'figure.subplot.bottom' : .125 | ||||
}, config=True, | ||||
help="""Subset of matplotlib rcParams that should be different for the | ||||
inline backend.""" | ||||
) | ||||
MinRK
|
r5292 | |||
MinRK
|
r3973 | figure_format = CaselessStrEnum(['svg', 'png'], default_value='png', config=True, | ||
help="The image format for figures with the inline backend.") | ||||
MinRK
|
r4005 | def _figure_format_changed(self, name, old, new): | ||
MinRK
|
r3973 | if self.shell is None: | ||
return | ||||
else: | ||||
select_figure_format(self.shell, new) | ||||
MinRK
|
r5087 | |||
close_figures = CBool(True, config=True, | ||||
help="""Close all figures at the end of each cell. | ||||
When True, ensures that each cell starts with no active figures, but it | ||||
also means that one must keep track of references in order to edit or | ||||
redraw figures in subsequent cells. This mode is ideal for the notebook, | ||||
where residual plots from other cells might be surprising. | ||||
When False, one must call figure() to create new figures. This means | ||||
that gcf() and getfigs() can reference figures created in other cells, | ||||
and the active figure can continue to be edited with pylab/pyplot | ||||
methods that reference the current active figure. This mode facilitates | ||||
iterative editing of figures, and behaves most consistently with | ||||
other matplotlib backends, but figure barriers between cells must | ||||
be explicit. | ||||
""") | ||||
MinRK
|
r3973 | |||
shell = Instance('IPython.core.interactiveshell.InteractiveShellABC') | ||||
epatters
|
r2756 | |||
Fernando Perez
|
r2890 | #----------------------------------------------------------------------------- | ||
# Functions | ||||
#----------------------------------------------------------------------------- | ||||
epatters
|
r2756 | |||
MinRK
|
r5087 | def show(close=None): | ||
"""Show all figures as SVG/PNG payloads sent to the IPython clients. | ||||
Fernando Perez
|
r2987 | |||
Parameters | ||||
---------- | ||||
close : bool, optional | ||||
If true, a ``plt.close('all')`` call is automatically issued after | ||||
MinRK
|
r5086 | sending all the figures. If this is set, the figures will entirely | ||
Brian Granger
|
r3280 | removed from the internal list of figures. | ||
epatters
|
r2756 | """ | ||
MinRK
|
r5087 | if close is None: | ||
MinRK
|
r5227 | close = InlineBackend.instance().close_figures | ||
MinRK
|
r5346 | try: | ||
for figure_manager in Gcf.get_all_fig_managers(): | ||||
send_figure(figure_manager.canvas.figure) | ||||
finally: | ||||
show._to_draw = [] | ||||
if close: | ||||
matplotlib.pyplot.close('all') | ||||
MinRK
|
r5087 | |||
Fernando Perez
|
r2987 | |||
Brian Granger
|
r3280 | |||
Fernando Perez
|
r2987 | # This flag will be reset by draw_if_interactive when called | ||
show._draw_called = False | ||||
MinRK
|
r5086 | # list of figures to draw when flush_figures is called | ||
show._to_draw = [] | ||||
Fernando Perez
|
r2987 | |||
def draw_if_interactive(): | ||||
""" | ||||
Is called after every pylab drawing command | ||||
""" | ||||
Fernando Perez
|
r6517 | # signal that the current active figure should be sent at the end of | ||
# execution. Also sets the _draw_called flag, signaling that there will be | ||||
# something to send. At the end of the code execution, a separate call to | ||||
# flush_figures() will act upon these values | ||||
MinRK
|
r5086 | fig = Gcf.get_active().canvas.figure | ||
Fernando Perez
|
r6517 | |||
# Hack: matplotlib FigureManager objects in interacive backends (at least | ||||
# in some of them) monkeypatch the figure object and add a .show() method | ||||
# to it. This applies the same monkeypatch in order to support user code | ||||
# that might expect `.show()` to be part of the official API of figure | ||||
# objects. | ||||
# For further reference: | ||||
# https://github.com/ipython/ipython/issues/1612 | ||||
# https://github.com/matplotlib/matplotlib/issues/835 | ||||
MinRK
|
r5086 | |||
Fernando Perez
|
r6517 | if not hasattr(fig, 'show'): | ||
# Queue up `fig` for display | ||||
fig.show = lambda *a: send_figure(fig) | ||||
# If matplotlib was manually set to non-interactive mode, this function | ||||
# should be a no-op (otherwise we'll generate duplicate plots, since a user | ||||
# who set ioff() manually expects to make separate draw/show calls). | ||||
if not matplotlib.is_interactive(): | ||||
return | ||||
MinRK
|
r5086 | # ensure current figure will be drawn, and each subsequent call | ||
# of draw_if_interactive() moves the active figure to ensure it is | ||||
# drawn last | ||||
try: | ||||
show._to_draw.remove(fig) | ||||
except ValueError: | ||||
# ensure it only appears in the draw list once | ||||
pass | ||||
Fernando Perez
|
r6517 | # Queue up the figure for drawing in next show() call | ||
MinRK
|
r5086 | show._to_draw.append(fig) | ||
Fernando Perez
|
r2987 | show._draw_called = True | ||
Fernando Perez
|
r6517 | |||
MinRK
|
r3973 | def flush_figures(): | ||
MinRK
|
r5086 | """Send all figures that changed | ||
Fernando Perez
|
r2987 | |||
This is meant to be called automatically and will call show() if, during | ||||
prior code execution, there had been any calls to draw_if_interactive. | ||||
MinRK
|
r5735 | |||
This function is meant to be used as a post_execute callback in IPython, | ||||
so user-caused errors are handled with showtraceback() instead of being | ||||
allowed to raise. If this function is not called from within IPython, | ||||
then these exceptions will raise. | ||||
Fernando Perez
|
r2987 | """ | ||
MinRK
|
r5086 | if not show._draw_called: | ||
return | ||||
MinRK
|
r5087 | |||
MinRK
|
r5227 | if InlineBackend.instance().close_figures: | ||
MinRK
|
r5087 | # ignore the tracking, just draw and close all figures | ||
MinRK
|
r5735 | try: | ||
return show(True) | ||||
except Exception as e: | ||||
# safely show traceback if in IPython, else raise | ||||
try: | ||||
MinRK
|
r5736 | get_ipython | ||
MinRK
|
r5735 | except NameError: | ||
raise e | ||||
MinRK
|
r5736 | else: | ||
get_ipython().showtraceback() | ||||
return | ||||
MinRK
|
r5346 | try: | ||
# exclude any figures that were closed: | ||||
active = set([fm.canvas.figure for fm in Gcf.get_all_fig_managers()]) | ||||
for fig in [ fig for fig in show._to_draw if fig in active ]: | ||||
MinRK
|
r5735 | try: | ||
send_figure(fig) | ||||
except Exception as e: | ||||
# safely show traceback if in IPython, else raise | ||||
try: | ||||
MinRK
|
r5736 | get_ipython | ||
MinRK
|
r5735 | except NameError: | ||
raise e | ||||
MinRK
|
r5736 | else: | ||
get_ipython().showtraceback() | ||||
break | ||||
MinRK
|
r5346 | finally: | ||
# clear flags for next round | ||||
show._to_draw = [] | ||||
show._draw_called = False | ||||
Brian Granger
|
r3280 | |||
MinRK
|
r3973 | def send_figure(fig): | ||
Fernando Perez
|
r6517 | """Draw the given figure and send it as a PNG payload. | ||
Brian Granger
|
r3280 | """ | ||
MinRK
|
r5227 | fmt = InlineBackend.instance().figure_format | ||
MinRK
|
r3973 | data = print_figure(fig, fmt) | ||
MinRK
|
r5735 | # print_figure will return None if there's nothing to draw: | ||
if data is None: | ||||
return | ||||
MinRK
|
r3973 | mimetypes = { 'png' : 'image/png', 'svg' : 'image/svg+xml' } | ||
mime = mimetypes[fmt] | ||||
Fernando Perez
|
r3731 | # flush text streams before sending figures, helps a little with output | ||
# synchronization in the console (though it's a bandaid, not a real sln) | ||||
sys.stdout.flush(); sys.stderr.flush() | ||||
Brian Granger
|
r3280 | publish_display_data( | ||
MinRK
|
r3973 | 'IPython.zmq.pylab.backend_inline.send_figure', | ||
{mime : data} | ||||
Brian Granger
|
r3280 | ) | ||