# Repository: https://gitlab.com/quantify-os/quantify-scheduler
# Licensed according to the LICENCE file on the main branch
"""
Module containing functions for drawing pulse schemes and circuit diagrams
using matplotlib.
"""
# pylint: disable=too-many-arguments
from __future__ import annotations
import logging
from typing import List, Optional, Tuple, Union
import matplotlib.patches
import matplotlib.pyplot as plt
import numpy as np
from matplotlib.axes import Axes
# For type hints, import modules to avoid circular dependencies
from matplotlib.figure import Figure
from quantify_scheduler.schedules._visualization import constants
[docs]logger = logging.getLogger(__name__)
[docs]def new_pulse_fig(
figsize: Optional[Tuple[int, int]] = None, ax: Optional[Axes] = None
) -> Tuple[Figure, Union[Axes, List[Axes]]]:
"""
Open a new figure and configure it to plot pulse schemes.
Parameters
----------
figsize
ax
Axis to use for plotting. If ``None``, then creates a new one.
Returns
-------
:
Tuple of figure handle and axis handle.
"""
if ax is None:
fig, ax = plt.subplots(1, 1, figsize=figsize, frameon=False)
else:
fig = None
ax.get_xaxis().set_visible(False)
ax.get_yaxis().set_ticklabels([])
ax.spines["top"].set_visible(False)
ax.spines["right"].set_visible(False)
ax.spines["bottom"].set_visible(False)
ax.spines["left"].set_visible(False)
if fig is not None:
fig.patch.set_alpha(0)
ax.axhline(0, color="0.75")
return fig, ax
[docs]def new_pulse_subplot(fig: Figure, *args, **kwargs) -> Axes:
"""
Add a new subplot configured for plotting pulse schemes to a figure.
All `*args` and `**kwargs` are passed to fig.add_subplot.
Parameters
----------
fig :
Returns
-------
:
"""
ax = fig.add_subplot(*args, **kwargs)
ax.axis("off")
fig.subplots_adjust(bottom=0, top=1, left=0, right=1)
ax.axhline(0, color="0.75")
return ax
[docs]def mwPulse(
ax: Axes,
pos: float,
y_offs: float = 0.0,
width: float = 1.5,
amp: float = 1,
label: Optional[str] = None,
phase=0,
label_height: float = 1.3,
color: str = constants.COLOR_ORANGE,
modulation: str = "normal",
**plot_kws,
) -> float:
"""
Draw a microwave pulse: Gaussian envelope with modulation.
Parameters
----------
ax :
pos :
y_offs :
width :
amp :
label :
label_height :
color :
modulation :
Returns
-------
:
"""
x = np.linspace(pos, pos + width, 100)
envPos = amp * np.exp(-((x - (pos + width / 2)) ** 2) / (width / 4) ** 2)
envNeg = -amp * np.exp(-((x - (pos + width / 2)) ** 2) / (width / 4) ** 2)
if modulation == "normal":
mod = envPos * np.sin(2 * np.pi * 3 / width * x + phase)
elif modulation == "high":
mod = envPos * np.sin(5 * np.pi * 3 / width * x + phase)
else:
raise ValueError()
ax.plot(x, envPos + y_offs, "--", color=color, **plot_kws)
ax.plot(x, envNeg + y_offs, "--", color=color, **plot_kws)
ax.plot(x, mod + y_offs, "-", color=color, **plot_kws)
if label is not None:
ax.text(
pos + width / 2,
label_height,
label,
horizontalalignment="right",
color=color,
).set_clip_on(True)
return pos + width
[docs]def fluxPulse(
ax: Axes,
pos: float,
y_offs: float = 0.0,
width: float = 2.5,
s: float = 0.1,
amp: float = 1.5,
label: Optional[str] = None,
label_height: float = 1.7,
color: str = constants.COLOR_ORANGE,
**plot_kws,
) -> float:
"""
Draw a smooth flux pulse, where the rising and falling edges are given by
Fermi-Dirac functions.
Parameters
----------
ax :
pos :
y_offs :
width :
s :
smoothness of edge
amp :
label :
label_height :
color :
Returns
-------
:
"""
x = np.linspace(pos, pos + width, 100)
y = amp / (
(np.exp(-(x - (pos + 5.5 * s)) / s) + 1)
* (np.exp((x - (pos + width - 5.5 * s)) / s) + 1)
)
ax.fill_between(x, y + y_offs, y_offs, color=color, alpha=0.3)
ax.plot(x, y + y_offs, color=color, **plot_kws)
if label is not None:
ax.text(
pos + width / 2,
label_height,
label,
horizontalalignment="center",
color=color,
).set_clip_on(True)
return pos + width
[docs]def ramZPulse(
ax: Axes,
pos: float,
y_offs: float = 0.0,
width: float = 2.5,
s: float = 0.1,
amp: float = 1.5,
sep: float = 1.5,
color: str = constants.COLOR_ORANGE,
) -> float:
"""
Draw a Ram-Z flux pulse, i.e. only part of the pulse is shaded, to indicate
cutting off the pulse at some time.
Parameters
----------
ax :
pos :
y_offs :
width :
s :
amp :
sep :
color :
Returns
-------
:
"""
xLeft = np.linspace(pos, pos + sep, 100)
xRight = np.linspace(pos + sep, pos + width, 100)
xFull = np.concatenate((xLeft, xRight))
y = amp / (
(np.exp(-(xFull - (pos + 5.5 * s)) / s) + 1)
* (np.exp((xFull - (pos + width - 5.5 * s)) / s) + 1)
)
yLeft = y[: len(xLeft)]
ax.fill_between(
xLeft, yLeft + y_offs, y_offs, alpha=0.3, color=color, linewidth=0.0
)
ax.plot(xFull, y + y_offs, color=color)
return pos + width
[docs]def interval(
ax: Axes,
start: float,
stop: float,
y_offs: float = 0.0,
height: float = 1.5,
label: Optional[str] = None,
label_height: Optional[str] = None,
vlines: bool = True,
color: str = "k",
arrowstyle: str = "<|-|>",
**plot_kws,
) -> None:
"""
Draw an arrow to indicate an interval.
Parameters
----------
ax :
start :
stop :
y_offs :
height :
label :
label_height :
vlines :
color :
arrowstyle :
Returns
-------
:
"""
if label_height is None:
label_height = height + 0.2
arrow = matplotlib.patches.FancyArrowPatch(
posA=(start, height + y_offs),
posB=(stop, height + y_offs),
arrowstyle=arrowstyle,
color=color,
mutation_scale=7,
**plot_kws,
)
ax.add_patch(arrow)
if vlines:
ax.plot(
[start, start], [0 + y_offs, height + y_offs], "--", color=color, **plot_kws
)
ax.plot(
[stop, stop], [0 + y_offs, height + y_offs], "--", color=color, **plot_kws
)
if label is not None:
ax.text(
(start + stop) / 2, label_height + y_offs, label, color=color, ha="center"
).set_clip_on(True)
[docs]def meter(
ax: Axes,
x0: float,
y0: float,
y_offs: float = 0.0,
width: float = 1.1,
height: float = 0.8,
color: str = "black",
framewidth: float = 0.0,
fillcolor: Optional[str] = None,
) -> None:
"""
Draws a measurement meter on the specified position.
Parameters
----------
ax :
x0 :
y0 :
y_offs :
width :
height :
color :
framewidth:
fillcolor :
Returns
-------
:
"""
if fillcolor is None:
fill = False
else:
fill = True
p1 = matplotlib.patches.Rectangle(
(x0 - width / 2, y0 - height / 2 + y_offs),
width,
height,
facecolor=fillcolor,
edgecolor=color,
fill=fill,
zorder=5,
linewidth=framewidth,
)
ax.add_patch(p1)
p0 = matplotlib.patches.Wedge(
(x0, y0 - height / constants.METER_WEDGE_HEIGHT_SCALING + y_offs),
constants.METER_WEDGE_RADIUS,
theta1=constants.METER_WEDGE_ANGLE,
theta2=180 - constants.METER_WEDGE_ANGLE,
color=color,
lw=1.5,
width=0.01,
zorder=5,
)
ax.add_patch(p0)
arrow_len = height / 2.0
ax.arrow(
x0,
y0 - height / constants.METER_ARROW_HEIGHT_SCALING + y_offs,
dx=arrow_len * np.cos(np.deg2rad(constants.METER_ARROW_ANGLE)),
dy=arrow_len * np.sin(np.deg2rad(constants.METER_ARROW_ANGLE)),
width=0.025,
color=color,
zorder=5,
)
[docs]def box_text(
ax: Axes,
x0: float,
y0: float,
text: str = "",
width: float = 1.1,
height: float = 0.8,
color: str = "black",
fillcolor: Optional[str] = None,
textcolor: str = "black",
fontsize: Optional[int] = None,
) -> None:
"""
Draws a box filled with text at the specified position.
Parameters
----------
ax :
x0 :
y0 :
text :
width :
height :
color :
fillcolor :
textcolor :
fontsize :
Returns
-------
:
"""
if fillcolor is None:
fill = False
else:
fill = True
p1 = matplotlib.patches.Rectangle(
(x0 - width / 2, y0 - height / 2),
width,
height,
facecolor=fillcolor,
edgecolor=color,
fill=fill,
linewidth=0,
zorder=5,
)
ax.add_patch(p1)
ax.text(
x0,
y0,
text,
ha="center",
va="center",
zorder=6,
size=fontsize,
color=textcolor,
).set_clip_on(True)