Source code for edifice.extra.matplotlib_figure
from __future__ import annotations
import typing as tp
from edifice.base_components.base_components import CommandType, QtWidgetElement
from edifice.qt import QT_VERSION
if QT_VERSION == "PyQt6":
pass
else:
pass
from matplotlib.axes import Axes
from matplotlib.backends.backend_qtagg import FigureCanvasQTAgg
from matplotlib.figure import Figure
if tp.TYPE_CHECKING:
from matplotlib.backend_bases import MouseEvent
[docs]
class MatplotlibFigure(QtWidgetElement):
"""
A **matplotlib** `Figure <https://matplotlib.org/stable/api/figure_api.html#matplotlib.figure.Figure>`_.
Requires `matplotlib <https://matplotlib.org/stable/>`_.
.. rubric:: Props
All **props** from :class:`edifice.QtWidgetElement` plus:
Args:
plot_fun:
Function which takes **matplotlib**
`Axes <https://matplotlib.org/stable/api/axes_api.html>`_
and calls
`Axes.plot <https://matplotlib.org/stable/api/_as_gen/matplotlib.axes.Axes.plot.html>`_.
on_figure_mouse_move:
Handler for mouse move
`MouseEvent <https://matplotlib.org/stable/api/backend_bases_api.html#matplotlib.backend_bases.MouseEvent>`_.
.. rubric:: Usage
.. code-block:: python
from matplotlib.axes import Axes
import numpy as np
from edifice.extra import MatplotlibFigure
def plot_fun(ax:Axes):
time_range = np.linspace(-10, 10, num=120)
ax.plot(time_range, np.sin(time_range))
MatplotlibFigure(plot_fun=plot_fun)
"""
[docs]
def __init__(
self,
plot_fun: tp.Callable[[Axes], None],
on_figure_mouse_move: tp.Callable[[MouseEvent], None] | None = None,
**kwargs,
):
super().__init__(**kwargs)
self._register_props(
{
"plot_fun": plot_fun,
"on_figure_mouse_move": on_figure_mouse_move,
},
)
self.underlying: FigureCanvasQTAgg | None = None
self.subplots: Axes | None = None
self.current_plot_fun: tp.Callable[[Axes], None] | None = None
self.on_mouse_move_connect_id: int | None = None
def _qt_update_commands(self, children, newprops):
if self.underlying is None:
# Default to maximum figsize https://matplotlib.org/stable/api/figure_api.html#matplotlib.figure.figaspect
# Constrain the Figure by putting it in a smaller View, it will resize itself correctly.
self.underlying = FigureCanvasQTAgg(Figure(figsize=(16.0, 16.0)))
self.subplots = tp.cast(Axes, self.underlying.figure.subplots()) # TODO is this cast valid?
assert self.underlying is not None
assert self.subplots is not None
commands = super()._qt_update_commands_super(children, newprops, self.underlying, None)
if "plot_fun" in newprops:
def _command_plot_fun(self):
self.current_plot_fun = tp.cast(tp.Callable[[Axes], None], self.props.plot_fun)
self.subplots.clear()
self.current_plot_fun(self.subplots)
self.underlying.draw()
# alternately we could do draw_idle() here, but I don't think it's
# any better and it messes up the mouse events.
commands.append(CommandType(_command_plot_fun, self))
if "on_figure_mouse_move" in newprops:
def _command_mouse_move(self):
if self.on_mouse_move_connect_id is not None:
self.underlying.mpl_disconnect(self.on_mouse_move_connect_id)
if newprops["on_figure_mouse_move"] is not None:
self.on_mouse_move_connect_id = self.underlying.mpl_connect(
"motion_notify_event",
newprops["on_figure_mouse_move"],
)
else:
self.on_mouse_move_connect_id = None
commands.append(CommandType(_command_mouse_move, self))
return commands