backend_inline.py
163 lines
| 6.1 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 | ||
MinRK
|
r3973 | from IPython.lib.pylabtools import print_figure, select_figure_format | ||
MinRK
|
r5087 | from IPython.utils.traitlets import Dict, Instance, CaselessStrEnum, CBool | ||
MinRK
|
r3973 | #----------------------------------------------------------------------------- | ||
# Configurable for inline backend options | ||||
#----------------------------------------------------------------------------- | ||||
class InlineBackendConfig(SingletonConfigurable): | ||||
"""An object to store configuration of the inline backend.""" | ||||
# The typical default figure size is too large for inline use, | ||||
# so we shrink the figure size to 6x4, and tweak fonts to | ||||
# make that fit. This is configurable via Global.pylab_inline_rc, | ||||
# or rather it will be once the zmq kernel is hooked up to | ||||
# the config system. | ||||
rc = Dict({'figure.figsize': (6.0,4.0), | ||||
# 12pt labels get cutoff on 6x4 logplots, so use 10pt. | ||||
'font.size': 10, | ||||
# 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.""" | ||||
) | ||||
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: | ||
close = InlineBackendConfig.instance().close_figures | ||||
Fernando Perez
|
r2890 | for figure_manager in Gcf.get_all_fig_managers(): | ||
MinRK
|
r3973 | send_figure(figure_manager.canvas.figure) | ||
Fernando Perez
|
r2987 | if close: | ||
matplotlib.pyplot.close('all') | ||||
MinRK
|
r5086 | show._to_draw = [] | ||
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 | ||||
""" | ||||
MinRK
|
r5086 | # 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 | ||||
fig = Gcf.get_active().canvas.figure | ||||
# 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 | ||||
show._to_draw.append(fig) | ||||
Fernando Perez
|
r2987 | show._draw_called = True | ||
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
|
r5086 | if not show._draw_called: | ||
return | ||||
MinRK
|
r5087 | |||
if InlineBackendConfig.instance().close_figures: | ||||
# ignore the tracking, just draw and close all figures | ||||
return show(True) | ||||
MinRK
|
r5086 | # 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 ]: | ||||
send_figure(fig) | ||||
# clear flags for next round | ||||
show._to_draw = [] | ||||
show._draw_called = False | ||||
Brian Granger
|
r3280 | |||
MinRK
|
r3973 | def send_figure(fig): | ||
"""Draw the current figure and send it as a PNG payload. | ||||
Brian Granger
|
r3280 | """ | ||
Fernando Perez
|
r3731 | # For an empty figure, don't even bother calling figure_to_svg, to avoid | ||
# big blank spaces in the qt console | ||||
if not fig.axes: | ||||
return | ||||
MinRK
|
r3973 | fmt = InlineBackendConfig.instance().figure_format | ||
data = print_figure(fig, fmt) | ||||
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 | ) | ||