Source code for pennylane.drawer.mpldrawer
# Copyright 2018-2021 Xanadu Quantum Technologies Inc.
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
# http://www.apache.org/licenses/LICENSE-2.0
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""
This module contains the MPLDrawer class for creating circuit diagrams with matplotlib
"""
import warnings
from collections.abc import Iterable, Sequence
has_mpl = True
try:
import matplotlib.patheffects as path_effects
import matplotlib.pyplot as plt
from matplotlib import patches
except (ModuleNotFoundError, ImportError) as e: # pragma: no cover
has_mpl = False
# pylint: disable=too-many-positional-arguments
def _to_tuple(a):
"""Converts int or iterable to tuple"""
if a is None:
return tuple()
if isinstance(a, Iterable):
return tuple(a)
return (a,)
def _open_circ_options_process(options):
"""For use in both ``_ctrlo_circ`` and ``_target_x``."""
if options is None:
options = {}
new_options = options.copy()
if "color" in new_options:
new_options["facecolor"] = plt.rcParams["axes.facecolor"]
new_options["edgecolor"] = options["color"]
new_options["color"] = None
else:
new_options["edgecolor"] = plt.rcParams["lines.color"]
new_options["facecolor"] = plt.rcParams["axes.facecolor"]
if "linewidth" not in new_options:
new_options["linewidth"] = plt.rcParams["lines.linewidth"]
if "zorder" not in new_options:
new_options["zorder"] = 3
return new_options
# pylint: disable=too-many-instance-attributes, too-many-arguments
[docs]class MPLDrawer:
r"""Allows easy creation of graphics representing circuits with matplotlib
Args:
n_layers (int): the number of layers
wire_map (dict): the wires to be drawn. A dict mapping wire label to index (from top to bottom) in the figure
Keyword Args:
c_wires=0 (int): the number of classical wires to leave space for.
wire_options=None (dict): matplotlib configuration options for drawing the wire lines
figsize=None (Iterable): Allows users to specify the size of the figure manually. Defaults
to scale with the size of the circuit via ``n_layers`` and ``len(wire_map)``.
fig=None (matplotlib Figure): Allows users to specify the figure window to plot to.
**Example**
.. code-block:: python
drawer = qml.drawer.MPLDrawer(wire_map={i: i for i in range(5)}, n_layers=6)
drawer.label(["0", "a", r"$|\Psi\rangle$", r"$|\theta\rangle$", "aux"])
drawer.box_gate(layer=0, wires=[0, 1, 2, 3, 4], text="Entangling Layers")
drawer.box_gate(layer=1, wires=[0, 2, 3], text="U(θ)")
drawer.box_gate(layer=1, wires=4, text="Z")
drawer.SWAP(layer=2, wires=(3,4))
drawer.CNOT(layer=2, wires=(0, 2))
drawer.ctrl(layer=3, wires=[1, 3], control_values=[True, False])
drawer.box_gate(
layer=3, wires=2, text="H", box_options={"zorder": 4}, text_options={"zorder": 5}
)
drawer.ctrl(layer=4, wires=[1, 2])
drawer.measure(layer=5, wires=0)
drawer.fig.suptitle('My Circuit', fontsize='xx-large')
.. figure:: ../../_static/drawer/example_basic.png
:align: center
:width: 60%
:target: javascript:void(0);
.. details::
:title: Usage Details
**Matplotlib Integration**
This class relies on matplotlib. As such, users can extend this class via interacting with the figure
``drawer.fig`` and axes ``drawer.ax`` objects manually. For instance, the example circuit manipulates the
figure to set a title using ``drawer.fig.suptitle``. Users can save the image using ``plt.savefig`` or via
the figure method ``drawer.fig.savefig``.
As described in the next section, the figure supports both global styling and individual styling of
elements with matplotlib styles, configuration, and keywords.
**Formatting**
PennyLane has inbuilt styles for controlling the appearance of the circuit drawings.
All available styles can be determined by evaluating ``qml.drawer.available_styles()``.
Any available string can then be passed to ``qml.drawer.use_style``.
.. code-block:: python
qml.drawer.use_style('black_white')
.. figure:: ../../_static/drawer/black_white_style.png
:align: center
:width: 60%
:target: javascript:void(0);
You can also control the appearance with matplotlib's provided tools, see the
`matplotlib docs <https://matplotlib.org/stable/tutorials/introductory/customizing.html>`_ .
For example, we can customize ``plt.rcParams``:
.. code-block:: python
plt.rcParams['patch.facecolor'] = 'mistyrose'
plt.rcParams['patch.edgecolor'] = 'maroon'
plt.rcParams['text.color'] = 'maroon'
plt.rcParams['font.weight'] = 'bold'
plt.rcParams['patch.linewidth'] = 4
plt.rcParams['patch.force_edgecolor'] = True
plt.rcParams['lines.color'] = 'indigo'
plt.rcParams['lines.linewidth'] = 5
plt.rcParams['figure.facecolor'] = 'ghostwhite'
.. figure:: ../../_static/drawer/example_rcParams.png
:align: center
:width: 60%
:target: javascript:void(0);
You can also manually control the styles of individual plot elements via the drawer class.
All accept dictionaries of keyword-values pairs for matplotlib object
components. Acceptable keywords differ based on what's being drawn. For example, you cannot pass ``"fontsize"``
to the dictionary controlling how to format a rectangle. For the control-type gates ``CNOT`` and
``ctrl`` the options dictionary can only contain ``'linewidth'``, ``'color'``, or ``'zorder'`` keys.
This example demonstrates the different ways you can format the individual elements:
.. code-block:: python
wire_options = {"color": "indigo", "linewidth": 4}
drawer = MPLDrawer(wire_map={0: 0, 1: 1}, n_layers=4, wire_options=wire_options)
label_options = {"fontsize": "x-large", 'color': 'indigo'}
drawer.label(["0", "a"], text_options=label_options)
box_options = {'facecolor': 'lightcoral', 'edgecolor': 'maroon', 'linewidth': 5}
text_options = {'fontsize': 'xx-large', 'color': 'maroon'}
drawer.box_gate(layer=0, wires=0, text="Z", box_options=box_options, text_options=text_options)
swap_options = {'linewidth': 4, 'color': 'darkgreen'}
drawer.SWAP(layer=1, wires=(0, 1), options=swap_options)
ctrl_options = {'linewidth': 4, 'color': 'teal'}
drawer.CNOT(layer=2, wires=(0, 1), options=ctrl_options)
drawer.ctrl(layer=3, wires=(0, 1), options=ctrl_options)
measure_box = {'facecolor': 'white', 'edgecolor': 'indigo'}
measure_lines = {'edgecolor': 'indigo', 'facecolor': 'plum', 'linewidth': 2}
for wire in range(2):
drawer.measure(layer=4, wires=wire, box_options=measure_box, lines_options=measure_lines)
drawer.fig.suptitle('My Circuit', fontsize='xx-large')
.. figure:: ../../_static/drawer/example_formatted.png
:align: center
:width: 60%
:target: javascript:void(0);
**Positioning**
Each gate takes arguments in order of ``layer`` followed by ``wires``. These translate to ``x`` and
``y`` coordinates in the graph. Layer number (``x``) increases as you go right, and wire number
(``y``) increases as you go down; the y-axis is inverted. You can pass non-integer values to either keyword.
If you have a long label, the gate can span multiple layers and have extra width:
.. code-block:: python
drawer = MPLDrawer(2, {0:0, 1:1})
drawer.box_gate(layer=0, wires=1, text="X")
drawer.box_gate(layer=1, wires=1, text="Y")
# Gate between two layers
drawer.box_gate(layer=0.5, wires=0, text="Big Gate", extra_width=0.5)
.. figure:: ../../_static/drawer/float_layer.png
:align: center
:width: 60%
:target: javascript:void(0);
"""
_box_length = 0.75
"""The width/height of the rectangle drawn by ``box_gate``"""
_circ_rad = 0.3
"""The radius of CNOT's target symbol."""
_ctrl_rad = 0.1
"""The radius of the control-on-one solid circle."""
_octrl_rad = 0.1
"""The radius of the control-on-zero open circle."""
_swap_dx = 0.2
"""Half the width/height of the SWAP X-symbol."""
_fontsize = 14
"""The default fontsize."""
_pad = 0.2
"""Padding for FancyBboxPatch objects."""
_boxstyle = "round, pad=0.2"
"""Style for FancyBboxPatch objects."""
_notch_width = 0.04
"""The width of active wire notches."""
_notch_height = 0.25
"""The height of active wire notches."""
_notch_style = "round, pad=0.05"
"""Box style for active wire notches."""
_cond_shift = 0.03
"""The shift value from the centre axis for classical double-lines."""
_cwire_scaling = 0.25
"""The distance between successive control wires."""
def __init__(self, n_layers, wire_map, c_wires=0, wire_options=None, figsize=None, fig=None):
if not has_mpl: # pragma: no cover
raise ImportError(
"Module matplotlib is required for ``MPLDrawer`` class. "
"You can install matplotlib via \n\n pip install matplotlib"
)
self.n_layers = n_layers
self.n_wires = len(wire_map)
## Creating figure and ax
if figsize is None:
figheight = self.n_wires + self._cwire_scaling * c_wires + 1 + 0.5 * (c_wires > 0)
figsize = (self.n_layers + 3, figheight)
if fig is None:
self._fig = plt.figure(figsize=figsize)
else:
fig.clear()
fig.set_figwidth(figsize[0])
fig.set_figheight(figsize[1])
self._fig = fig
self._ax = self._fig.add_axes(
[0, 0, 1, 1],
xlim=(-2, self.n_layers + 1),
ylim=(-1, self.n_wires + self._cwire_scaling * c_wires + 0.5 * (c_wires > 0)),
xticks=[],
yticks=[],
)
self._ax.axis("off")
self._ax.invert_yaxis()
if wire_options is None:
wire_options = {}
# Separate global options from per wire options
global_options = {k: v for k, v in wire_options.items() if not isinstance(v, dict)}
wire_specific_options = {k: v for k, v in wire_options.items() if isinstance(v, dict)}
# Adding wire lines with individual styles based on wire_options
self._wire_lines = []
for wire_label, idx in wire_map.items():
specific_options = wire_specific_options.get(wire_label, {})
line_options = {**global_options, **specific_options}
# Create Line2D with the combined options
line = plt.Line2D(
(-1, self.n_layers),
(idx, idx),
zorder=1,
**line_options,
)
self._wire_lines.append(line)
for line in self._wire_lines:
self._ax.add_line(line)
@property
def fig(self):
"""Matplotlib figure"""
return self._fig
@property
def ax(self):
"""Matplotlib axes"""
return self._ax
@property
def fontsize(self):
"""Default fontsize for text. Defaults to 14."""
return self._fontsize
@fontsize.setter
def fontsize(self, value):
"""Set ``fontsize`` property as provided value."""
self._fontsize = value
[docs] def label(self, labels, text_options=None):
"""Label each wire.
Args:
labels (Iterable[str]): Iterable of labels for the wires
Keyword Args:
text_options (dict): any matplotlib keywords for a text object, such as font or size
**Example**
.. code-block:: python
drawer = MPLDrawer(wire_map={0:0, 1:1}, n_layers=1)
drawer.label(["a", "b"])
.. figure:: ../../_static/drawer/labels.png
:align: center
:width: 60%
:target: javascript:void(0);
You can also pass any
`Matplotlib Text keywords <https://matplotlib.org/stable/api/_as_gen/matplotlib.axes.Axes.text.html>`_
as a dictionary to the ``text_options`` keyword:
.. code-block:: python
drawer = MPLDrawer(wire_map={0:0, 1:1}, n_layers=1)
drawer.label(["a", "b"], text_options={"color": "indigo", "fontsize": "xx-large"})
.. figure:: ../../_static/drawer/labels_formatted.png
:align: center
:width: 60%
:target: javascript:void(0);
"""
if text_options is None:
text_options = {"ha": "center", "va": "center", "fontsize": self.fontsize}
for wire, ii_label in enumerate(labels):
self._ax.text(-1.5, wire, ii_label, **text_options)
[docs] def erase_wire(self, layer: int, wire: int, length: int) -> None:
"""Erases a portion of a wire by adding a rectangle that matches the background.
Args:
layer (int): starting x coordinate for erasing the wire
wire (int): y location to erase the wire from
length (float, int): horizontal distance from ``layer`` to erase the background.
"""
rect = patches.Rectangle(
(layer, wire - 0.1),
length,
0.2,
facecolor=plt.rcParams["figure.facecolor"],
edgecolor=plt.rcParams["figure.facecolor"],
zorder=1.1,
)
self.ax.add_patch(rect)
[docs] def box_gate(self, layer, wires, text="", box_options=None, text_options=None, **kwargs):
"""Draws a box and adds label text to its center.
Args:
layer (int): x coordinate for the box center
wires (Union[int, Iterable[int]]): y locations to include inside the box. Only min and max
of an Iterable affect the output
text (str): string to print at the box's center
Keyword Args:
box_options=None (dict): any matplotlib keywords for the ``plt.Rectangle`` patch
text_options=None (dict): any matplotlib keywords for the text
extra_width (float): extra box width
autosize (bool): whether to rotate and shrink text to fit within the box
active_wire_notches (bool): whether or not to add notches indicating active wires.
Defaults to ``True``.
**Example**
.. code-block:: python
drawer = MPLDrawer(wire_map={0:0, 1:1}, n_layers=1)
drawer.box_gate(layer=0, wires=(0, 1), text="CY")
.. figure:: ../../_static/drawer/box_gates.png
:align: center
:width: 60%
:target: javascript:void(0);
.. details::
:title: Usage Details
This method can accept two different sets of design keywords. ``box_options`` takes
`Rectangle keywords <https://matplotlib.org/stable/api/_as_gen/matplotlib.patches.Rectangle.html>`_
, and ``text_options`` accepts
`Matplotlib Text keywords <https://matplotlib.org/stable/api/_as_gen/matplotlib.axes.Axes.text.html>`_ .
.. code-block:: python
box_options = {'facecolor': 'lightcoral', 'edgecolor': 'maroon', 'linewidth': 5}
text_options = {'fontsize': 'xx-large', 'color': 'maroon'}
drawer = MPLDrawer(wire_map={0:0, 1:1}, n_layers=1)
drawer.box_gate(layer=0, wires=(0, 1), text="CY",
box_options=box_options, text_options=text_options)
.. figure:: ../../_static/drawer/box_gates_formatted.png
:align: center
:width: 60%
:target: javascript:void(0);
By default, text is rotated and/or shrunk to fit within the box. This behaviour can be turned off
with the ``autosize=False`` keyword.
.. code-block:: python
drawer = MPLDrawer(n_layers=4, wire_map={0:0, 1:1})
drawer.box_gate(layer=0, wires=0, text="A longer label")
drawer.box_gate(layer=0, wires=1, text="Label")
drawer.box_gate(layer=1, wires=(0,1), text="long multigate label")
drawer.box_gate(layer=3, wires=(0,1), text="Not autosized label", autosize=False)
.. figure:: ../../_static/drawer/box_gates_autosized.png
:align: center
:width: 60%
:target: javascript:void(0);
"""
extra_width = kwargs.get("extra_width", 0)
autosize = kwargs.get("autosize", True)
active_wire_notches = kwargs.get("active_wire_notches", True)
if box_options is None:
box_options = {}
if "zorder" not in box_options:
box_options["zorder"] = 2
new_text_options = {"zorder": 3, "ha": "center", "va": "center", "fontsize": self.fontsize}
if text_options is not None:
new_text_options.update(text_options)
wires = _to_tuple(wires)
box_min = min(wires)
box_max = max(wires)
box_center = (box_max + box_min) / 2.0
x_loc = layer - self._box_length / 2.0 - extra_width / 2.0 + self._pad
y_loc = box_min - self._box_length / 2.0 + self._pad
box_height = box_max - box_min + self._box_length - 2 * self._pad
box_width = self._box_length + extra_width - 2 * self._pad
box = patches.FancyBboxPatch(
(x_loc, y_loc),
box_width,
box_height,
boxstyle=self._boxstyle,
**box_options,
)
self._ax.add_patch(box)
text_obj = self._ax.text(
layer,
box_center,
text,
**new_text_options,
)
if active_wire_notches and (len(wires) != (box_max - box_min + 1)):
notch_options = box_options.copy()
notch_options["zorder"] += -1
for wire in wires:
self._add_notch(layer, wire, extra_width, notch_options)
if autosize:
margin = 0.1
max_width = box_width - margin + 2 * self._pad
# factor of 2 makes it look nicer
max_height = box_height - 2 * margin + 2 * self._pad
w, h = self._text_dims(text_obj)
# rotate the text
if (box_min != box_max) and (w > max_width) and (w > h):
text_obj.set_rotation(90)
w, h = self._text_dims(text_obj)
# shrink by decreasing the font size
current_fontsize = text_obj.get_fontsize()
for s in range(int(current_fontsize), 1, -1):
if (w < max_width) and (h < max_height):
break
text_obj.set_fontsize(s)
w, h = self._text_dims(text_obj)
def _add_notch(self, layer, wire, extra_width, box_options):
"""Add a wire used marker to both sides of a box.
Args:
layer (int): x coordinate for the box center
wire (int): y cordinate for the notches
extra_width (float): extra box width
box_options (dict): styling options
"""
y = wire - self._notch_height / 2
x1 = layer - self._box_length / 2.0 - extra_width / 2.0 - self._notch_width
x2 = layer + self._box_length / 2.0 + extra_width / 2.0
box1 = patches.FancyBboxPatch(
(x1, y),
self._notch_width,
self._notch_height,
boxstyle=self._notch_style,
**box_options,
)
self._ax.add_patch(box1)
box2 = patches.FancyBboxPatch(
(x2, y),
self._notch_width,
self._notch_height,
boxstyle=self._notch_style,
**box_options,
)
self._ax.add_patch(box2)
def _text_dims(self, text_obj):
"""Get width and height of text object in data coordinates.
See `this tutorial <https://matplotlib.org/stable/tutorials/advanced/transforms_tutorial.html>`_
for details on matplotlib coordinate systems.
If the renderered figure is resized, such as in a GUI display, rectangles and lines
are resized, but text stays the same size. Text objects rely on display coordinates, that wont shrink
as the figure is modified.
Args:
text_obj (matplotlib.text.Text): the matplotlib text object
Returns:
width (float): the width of the text in data coordinates
height (float): the height of the text in data coordinates
"""
renderer = self._fig.canvas.get_renderer()
# https://matplotlib.org/stable/api/_as_gen/matplotlib.artist.Artist.get_window_extent.html
# Quote: "Be careful when using this function, the results will not update if the artist
# window extent of the artist changes. "
# But I haven't encountered any issues yet and don't see a better solution
bbox = text_obj.get_window_extent(renderer)
corners = self._ax.transData.inverted().transform(bbox)
return abs(corners[1][0] - corners[0][0]), abs(corners[0][1] - corners[1][1])
[docs] def ctrl(self, layer, wires, wires_target=None, control_values=None, options=None):
"""Add an arbitrary number of control wires
Args:
layer (int): the layer to draw the object in
wires (Union[int, Iterable[int]]): set of wires to control on
Keyword Args:
wires_target=None (Union[int, Iterable[int]]): target wires. Used to determine min
and max wires for the vertical line
control_values=None (Union[bool, Iterable[bool]]): for each control wire, denotes whether to control
on ``False=0`` or ``True=1``
options=None (dict): Matplotlib keywords. The only supported keys are ``'color'``, ``'linewidth'``,
and ``'zorder'``.
**Example**
.. code-block:: python
drawer = MPLDrawer(wire_map={0:0, 1:1}, n_layers=3)
drawer.ctrl(layer=0, wires=0, wires_target=1)
drawer.ctrl(layer=1, wires=(0, 1), control_values=[0, 1])
options = {'color': "indigo", 'linewidth': 4}
drawer.ctrl(layer=2, wires=(0, 1), control_values=[1, 0], options=options)
.. figure:: ../../_static/drawer/ctrl.png
:align: center
:width: 60%
:target: javascript:void(0);
"""
if options is None:
options = {}
wires_ctrl = _to_tuple(wires)
wires_target = _to_tuple(wires_target)
if control_values is not None:
control_values = _to_tuple(control_values)
wires_all = wires_ctrl + wires_target
min_wire = min(wires_all)
max_wire = max(wires_all)
if len(wires_target) > 1:
min_target, max_target = min(wires_target), max(wires_target)
if any(min_target < w < max_target for w in wires_ctrl):
warnings.warn(
"Some control indicators are hidden behind an operator. Consider re-ordering "
"your circuit wires to ensure all control indicators are visible.",
UserWarning,
)
line = plt.Line2D((layer, layer), (min_wire, max_wire), **options)
self._ax.add_line(line)
if control_values is None:
for wire in wires_ctrl:
self._ctrl_circ(layer, wire, options=options)
else:
if len(control_values) != len(wires_ctrl):
raise ValueError("`control_values` must be the same length as `wires`")
for wire, control_on in zip(wires_ctrl, control_values):
if control_on:
self._ctrl_circ(layer, wire, options=options)
else:
self._ctrlo_circ(layer, wire, options=options)
def _ctrl_circ(self, layer, wires, options=None):
"""Draw a solid circle that indicates control on one.
Acceptable keys in options dictionary:
* zorder
* color
* linewidth
"""
if options is None:
options = {}
if "color" not in options:
options["color"] = plt.rcParams["lines.color"]
if "zorder" not in options:
options["zorder"] = 3
circ_ctrl = plt.Circle((layer, wires), radius=self._ctrl_rad, **options)
self._ax.add_patch(circ_ctrl)
def _ctrlo_circ(self, layer, wires, options=None):
"""Draw an open circle that indicates control on zero.
Acceptable keys in options dictionary:
* zorder
* color
* linewidth
"""
new_options = _open_circ_options_process(options)
circ_ctrlo = plt.Circle((layer, wires), radius=(self._octrl_rad), **new_options)
self._ax.add_patch(circ_ctrlo)
[docs] def CNOT(self, layer, wires, control_values=None, options=None):
"""Draws a CNOT gate.
Args:
layer (int): layer to draw in
control_values=None (Union[bool, Iterable[bool]]): for each control wire, denotes whether to control
on ``False=0`` or ``True=1``
wires (Union[int, Iterable[int]]): wires to use. Last wire is the target.
Keyword Args:
options=None: Matplotlib options. The only supported keys are ``'color'``, ``'linewidth'``,
and ``'zorder'``.
**Example**
.. code-block:: python
drawer = MPLDrawer(wire_map={0:0, 1:1}, n_layers=2)
drawer.CNOT(0, (0, 1))
options = {'color': 'indigo', 'linewidth': 4}
drawer.CNOT(1, (1, 0), options=options)
.. figure:: ../../_static/drawer/cnot.png
:align: center
:width: 60%
:target: javascript:void(0);
"""
self.ctrl(layer, wires[:-1], wires[-1], control_values=control_values, options=options)
self._target_x(layer, wires[-1], options=options)
def _target_x(self, layer, wires, options=None):
"""Draws the circle used to represent a CNOT's target
Args:
layer (int): layer to draw on
wires (int): wire to draw on
Keyword Args:
options=None (dict): Matplotlib keywords. The only supported keys are ``'color'``, ``'linewidth'``,
and ``'zorder'``.
"""
if options is None:
options = {}
new_options = _open_circ_options_process(options)
options["zorder"] = new_options["zorder"] + 1
target_circ = plt.Circle((layer, wires), radius=self._circ_rad, **new_options)
target_v = plt.Line2D(
(layer, layer), (wires - self._circ_rad, wires + self._circ_rad), **options
)
target_h = plt.Line2D(
(layer - self._circ_rad, layer + self._circ_rad), (wires, wires), **options
)
self._ax.add_patch(target_circ)
self._ax.add_line(target_v)
self._ax.add_line(target_h)
[docs] def SWAP(self, layer, wires, options=None):
"""Draws a SWAP gate
Args:
layer (int): layer to draw on
wires (Tuple[int, int]): two wires the SWAP acts on
Keyword Args:
options=None (dict): matplotlib keywords for ``Line2D`` objects
**Example**
The ``options`` keyword can accept any
`Line2D compatible keywords <https://matplotlib.org/stable/api/_as_gen/matplotlib.lines.Line2D.html#matplotlib.lines.Line2D>`_
in a dictionary.
.. code-block:: python
drawer = MPLDrawer(wire_map={0:0, 1:1}, n_layers=2)
drawer.SWAP(0, (0, 1))
swap_options = {"linewidth": 2, "color": "indigo"}
drawer.SWAP(1, (0, 1), options=swap_options)
.. figure:: ../../_static/drawer/SWAP.png
:align: center
:width: 60%
:target: javascript:void(0);
"""
if options is None:
options = {}
line = plt.Line2D((layer, layer), wires, **options)
self._ax.add_line(line)
for wire in wires:
self._swap_x(layer, wire, options)
def _swap_x(self, layer, wire, options=None):
"""Draw an x such as used in drawing a swap gate
Args:
layer (int): layer to draw on
wires (int): wire to draw on
Keyword Args:
options=None (dict): matplotlib keywords for ``Line2D`` objects
"""
if options is None:
options = {}
if "zorder" not in options:
options["zorder"] = 2
l1 = plt.Line2D(
(layer - self._swap_dx, layer + self._swap_dx),
(wire - self._swap_dx, wire + self._swap_dx),
**options,
)
l2 = plt.Line2D(
(layer - self._swap_dx, layer + self._swap_dx),
(wire + self._swap_dx, wire - self._swap_dx),
**options,
)
self._ax.add_line(l1)
self._ax.add_line(l2)
[docs] def measure(self, layer, wires, text=None, box_options=None, lines_options=None):
"""Draw a Measurement graphic at designated layer, wire combination.
Args:
layer (int): layer to draw on
wires (int): wire to draw on
Keyword Args:
text=None (str): an annotation for the lower right corner.
box_options=None (dict): dictionary to format a matplotlib rectangle
lines_options=None (dict): dictionary to format matplotlib arc and arrow
**Example**
This method accepts two different formatting dictionaries. ``box_options`` edits the rectangle
while ``lines_options`` edits the arc and arrow.
.. code-block:: python
drawer = MPLDrawer(wire_map={0:0, 1:1}, n_layers=1)
drawer.measure(layer=0, wires=0)
measure_box = {'facecolor': 'white', 'edgecolor': 'indigo'}
measure_lines = {'edgecolor': 'indigo', 'facecolor': 'plum', 'linewidth': 2}
drawer.measure(layer=0, wires=1, box_options=measure_box, lines_options=measure_lines)
.. figure:: ../../_static/drawer/measure.png
:align: center
:width: 60%
:target: javascript:void(0);
"""
if box_options is None:
box_options = {}
if "zorder" not in box_options:
box_options["zorder"] = 2
if lines_options is None:
lines_options = {}
if "zorder" not in lines_options:
lines_options["zorder"] = 3
if not isinstance(wires, Sequence):
wires = (wires,)
wires = tuple(self._y(w) for w in wires)
box_min = min(wires)
box_max = max(wires)
box_center = (box_max + box_min) / 2.0
x_loc = layer - self._box_length / 2.0 + self._pad
y_loc = box_min - self._box_length / 2.0 + self._pad
box = patches.FancyBboxPatch(
(x_loc, y_loc),
self._box_length - 2 * self._pad,
box_max - box_min + self._box_length - 2 * self._pad,
boxstyle=self._boxstyle,
**box_options,
)
self._ax.add_patch(box)
arc = patches.Arc(
(layer, box_center + 0.15 * self._box_length),
0.6 * self._box_length,
0.55 * self._box_length,
theta1=180,
theta2=0,
**lines_options,
)
self._ax.add_patch(arc)
# can experiment with the specific numbers to make it look decent
arrow_start_x = layer - 0.15 * self._box_length
arrow_start_y = box_center + 0.3 * self._box_length
arrow_width = 0.3 * self._box_length
arrow_height = -0.5 * self._box_length
lines_options["zorder"] += 1
self.ax.arrow(
arrow_start_x,
arrow_start_y,
arrow_width,
arrow_height,
head_width=self._box_length / 8.0,
**lines_options,
)
if text:
self._ax.text(
layer + 0.05 * self._box_length,
box_center + 0.225,
text,
fontsize=(self.fontsize - 2),
)
def _y(self, wire):
"""Used for determining the correct y coordinate for classical wires.
Classical wires should be enumerated starting at the number of quantum wires the drawer has.
For example, if the drawer has ``3`` quantum wires, the first classical wire should be located at ``3``
which corresponds to a ``y`` coordinate of ``2.9``.
"""
if wire < self.n_wires:
return wire
return self.n_wires + self._cwire_scaling * (wire - self.n_wires)
[docs] def classical_wire(self, layers, wires) -> None:
"""Draw a classical control line.
Args:
layers: a list of x coordinates for the classical wire
wires: a list of y coordinates for the classical wire. Wire numbers
greater than the number of quantum wires will be scaled as classical wires.
"""
outer_stroke = path_effects.Stroke(
linewidth=5 * plt.rcParams["lines.linewidth"], foreground=plt.rcParams["lines.color"]
)
inner_stroke = path_effects.Stroke(
linewidth=3 * plt.rcParams["lines.linewidth"],
foreground=plt.rcParams["figure.facecolor"],
)
line = plt.Line2D(
layers, [self._y(w) for w in wires], path_effects=[outer_stroke, inner_stroke], zorder=1
)
self.ax.add_line(line)
[docs] def cwire_join(self, layer, wire, erase_right=False):
"""Erase the horizontal edges of an intersection between classical wires. By default, erases
only the left edge.
Args:
layer: the x-coordinate for the classical wire intersection
wire: the classical wire y-coordinate for the intersection
erase_right=False(bool): whether or not to erase the right side of the intersection
in addition to the left.
"""
xs = (layer - 0.2, layer + 0.2) if erase_right else (layer - 0.2, layer)
line = plt.Line2D(
xs,
(self._y(wire), self._y(wire)),
zorder=2,
color=plt.rcParams["figure.facecolor"],
linewidth=3 * plt.rcParams["lines.linewidth"], # match inner_stroke from classical_wire
)
self.ax.add_line(line)
[docs] def cond(self, layer, measured_layer, wires, wires_target, options=None):
"""Add classical communication double-lines for conditional operations
Args:
layer (int): the layer to draw vertical lines in, containing the target operation
measured_layer (int): the layer where the mid-circuit measurements are
wires (Union[int, Iterable[int]]): set of wires to control on
wires_target (Union[int, Iterable[int]]): target wires. Used to determine where to
terminate the vertical double-line
Keyword Args:
options=None (dict): Matplotlib keywords passed to ``plt.Line2D``
**Example**
.. code-block:: python
drawer = MPLDrawer(wire_map={0:0, 1:1, 2:2}, n_layers=4)
drawer.cond(layer=1, measured_layer=0, wires=[0], wires_target=[1])
options = {'color': "indigo", 'linewidth': 1.5}
drawer.cond(layer=3, measured_layer=2, wires=(1,), wires_target=(2,), options=options)
.. figure:: ../../_static/drawer/cond.png
:align: center
:width: 60%
:target: javascript:void(0);
"""
if options is None:
options = {}
wires_ctrl = _to_tuple(sorted(wires))
wires_target = _to_tuple(sorted(wires_target))
start_x = measured_layer + self._box_length / 2.0
lines = []
if wires_ctrl[-1] < wires_target[0]:
lines.extend(
(
# draw from top-most measurement to double-elbow
plt.Line2D(
(start_x, layer + self._cond_shift),
(wires_ctrl[0] - self._cond_shift,) * 2,
**options,
),
plt.Line2D(
(start_x, layer - self._cond_shift),
(wires_ctrl[0] + self._cond_shift,) * 2,
**options,
),
# draw vertical lines that reach the target operation
plt.Line2D(
(layer + self._cond_shift,) * 2,
(wires_ctrl[0] - self._cond_shift, wires_target[0]),
**options,
),
plt.Line2D(
(layer - self._cond_shift,) * 2,
(wires_ctrl[-1] + self._cond_shift, wires_target[0]),
**options,
),
)
)
for prev_idx, next_wire in enumerate(wires_ctrl[1:]):
# draw ⅃ for every wire but the first one
# ‾
lines.extend(
(
plt.Line2D(
(layer - self._cond_shift,) * 2,
(wires_ctrl[prev_idx] + self._cond_shift, next_wire - self._cond_shift),
**options,
),
plt.Line2D(
(start_x, layer - self._cond_shift),
(next_wire - self._cond_shift,) * 2,
**options,
),
plt.Line2D(
(start_x, layer - self._cond_shift),
(next_wire + self._cond_shift,) * 2,
**options,
),
)
)
elif wires_target[-1] < wires_ctrl[0]:
lines.extend(
(
# draw from bottom-most measurement to double-elbow
plt.Line2D(
(start_x, layer + self._cond_shift),
(wires_ctrl[-1] + self._cond_shift,) * 2,
**options,
),
plt.Line2D(
(start_x, layer - self._cond_shift),
(wires_ctrl[-1] - self._cond_shift,) * 2,
**options,
),
# draw vertical lines that reach the target operation
plt.Line2D(
(layer + self._cond_shift,) * 2,
(wires_ctrl[-1] + self._cond_shift, wires_target[-1]),
**options,
),
plt.Line2D(
(layer - self._cond_shift,) * 2,
(wires_ctrl[0] - self._cond_shift, wires_target[-1]),
**options,
),
)
)
for wire_idx, ctrl_wire in enumerate(wires_ctrl[:-1]):
# draw _ for every wire but the first one
# ‾|
lines.extend(
(
plt.Line2D(
(layer - self._cond_shift,) * 2,
(
ctrl_wire + self._cond_shift,
wires_ctrl[wire_idx + 1] - self._cond_shift,
),
**options,
),
plt.Line2D(
(start_x, layer - self._cond_shift),
(ctrl_wire - self._cond_shift,) * 2,
**options,
),
plt.Line2D(
(start_x, layer - self._cond_shift),
(ctrl_wire + self._cond_shift,) * 2,
**options,
),
)
)
else:
raise ValueError(
"Cannot draw interspersed mid-circuit measurements and conditional operations. "
"Consider providing a wire order such that all measurement wires precede all "
"wires for the operator being controlled, or vice versa."
)
for line in lines:
self._ax.add_line(line)
[docs] def crop_wire_labels(self):
"""Crop away the wire labels and resize figure accordingly."""
xlim = self._ax.get_xlim()
self._ax.set_xlim((-1, xlim[1]))
factor = (self.n_layers + 2) / (self.n_layers + 3)
self._fig.set_figwidth(self._fig.get_figwidth() * factor)
_modules/pennylane/drawer/mpldrawer
Download Python script
Download Notebook
View on GitHub