import matplotlib.pyplot as plt
from matplotlib.patches import Rectangle, FancyBboxPatch, Circle
import qutip
from qutip_qip.circuit import QubitCircuit
from qutip_qip.operations import Gate
In [97]:
In [98]:
= 0.7
WIRE_SEP = 0.6
LAYER_SEP = 0.3
PAD_LABEL = 0.2
GATE_HEIGHT = 0.2
GATE_WIDTH = (LAYER_SEP - GATE_WIDTH)/2
GATE_PAD = 10 FONT_SIZE
In [99]:
def draw_wire(ax, n_wires):
=2
n_layers= [[(0, n_layers*LAYER_SEP) , (i*WIRE_SEP, i*WIRE_SEP) ] for i in range(n_wires)]
wires
for wire in wires:
= plt.Line2D(xdata=wire[0], ydata=wire[1], color='black', zorder=1)
line ax.add_line(line)
In [100]:
def extend_wire(ax, n_wires, old_layers, add_layers):
= [[(old_layers * LAYER_SEP, (old_layers + add_layers) * LAYER_SEP), (i * WIRE_SEP, i * WIRE_SEP)] for i in range(n_wires)]
wires
for wire in wires:
= plt.Line2D(xdata=wire[0], ydata=wire[1], color='black', zorder=1)
line ax.add_line(line)
In [101]:
def add_labels(ax, n_wires, wire_labels=[]):
if wire_labels == []:
= [f"$q_{{{i}}}$" for i in range(n_wires)]
wire_labels
for i, label in enumerate(wire_labels):
-PAD_LABEL, i * WIRE_SEP, label, fontsize=FONT_SIZE, verticalalignment='center') ax.text(
In [102]:
def draw_control_node(ax, layer, target):
# make a circle patch
= Circle((layer * LAYER_SEP + LAYER_SEP / 2, target * WIRE_SEP), 0.05, facecolor='blue', zorder=2)
node ax.add_patch(node)
In [103]:
def draw_target_node(ax, layer, target):
= 0.12
TARGET_RADIUS
# Draw the target node as a circle
= Circle((layer * LAYER_SEP + LAYER_SEP / 2, target * WIRE_SEP), TARGET_RADIUS, facecolor='blue', zorder=2)
node
ax.add_patch(node)
# Draw plus sign
= plt.Line2D((layer * LAYER_SEP + LAYER_SEP / 2, layer * LAYER_SEP + LAYER_SEP / 2),
vertical * WIRE_SEP - TARGET_RADIUS/2, target * WIRE_SEP + TARGET_RADIUS/2),
(target =1.5, color='white', zorder=3)
linewidth
= plt.Line2D((layer * LAYER_SEP + LAYER_SEP / 2 - TARGET_RADIUS/2, layer * LAYER_SEP + LAYER_SEP / 2 + TARGET_RADIUS/2),
horizontal * WIRE_SEP, target * WIRE_SEP),
(target =1.5, color='white', zorder=3)
linewidth
ax.add_line(vertical) ax.add_line(horizontal)
In [104]:
# add SWAP GATE mark "X"" on the target qubit
def draw_swap_mark(ax, layer, wire):
# Draw the diagonal lines to form an "X"
= plt.Line2D(
dia_left * LAYER_SEP + LAYER_SEP / 2 - GATE_WIDTH / 3, layer * LAYER_SEP + LAYER_SEP / 2 + GATE_WIDTH / 3],
[layer * WIRE_SEP - GATE_HEIGHT / 2, wire * WIRE_SEP + GATE_HEIGHT / 2],
[wire ="blue", linewidth=1.5, zorder=3
color
)= plt.Line2D(
dia_right * LAYER_SEP + LAYER_SEP / 2 + GATE_WIDTH / 3, layer * LAYER_SEP + LAYER_SEP / 2 - GATE_WIDTH / 3],
[layer * WIRE_SEP - GATE_HEIGHT / 2, wire * WIRE_SEP + GATE_HEIGHT / 2],
[wire ="blue", linewidth=1.5, zorder=3
color
)
ax.add_line(dia_left)
ax.add_line(dia_right)
In [105]:
def draw_bridge(ax, layer, wire1, wire2):
= plt.Line2D(
bridge *LAYER_SEP + LAYER_SEP/2, layer*LAYER_SEP + LAYER_SEP/2],
[layer*WIRE_SEP, wire2*WIRE_SEP],
[wire1='blue', linewidth=1, zorder=2
color
)
ax.add_line(bridge)
In [106]:
def multiq_add_gate(ax, layer, gate):
# check if gate is a qutip Gate
if not isinstance(gate, Gate):
print("Gate is not a qutip Gate")
return
# check if gate has multiple targets
if gate.name == "CNOT":
0],)
draw_control_node(ax, layer, gate.controls[0])
draw_target_node(ax, layer, gate.targets[0], gate.controls[0])
draw_bridge(ax, layer, gate.targets[
if gate.name == "SWAP":
0])
draw_swap_mark(ax, layer, gate.targets[1])
draw_swap_mark(ax, layer, gate.targets[0], gate.targets[1]) draw_bridge(ax, layer, gate.targets[
In [107]:
# adding gate, currently only works for qutip Gates
def add_gate(ax, layer, gate):
if isinstance(gate, Gate):
pass
# print("Gate is a qutip Gate")
else:
pass
# print("Gate is not a qutip Gate")
= f"${gate.name}$"
gate_label = gate.targets[0]
gate_wire
= FancyBboxPatch(
gate_patch * LAYER_SEP + GATE_PAD, gate_wire * WIRE_SEP - GATE_HEIGHT / 2),
(layer
GATE_WIDTH,
GATE_HEIGHT, ="round4",
boxstyle=0.3,
mutation_scale="aqua",
facecolor="aqua",
edgecolor=2
zorder
)
# gate_patch = Rectangle((layer*LAYER_SEP + GATE_PAD, gate_wire*WIRE_SEP - GATE_HEIGHT/2), GATE_WIDTH, GATE_HEIGHT, facecolor='aqua', zorder=2)
ax.add_patch(gate_patch)
# add gate label
*LAYER_SEP + GATE_PAD + GATE_WIDTH/2, gate_wire*WIRE_SEP, gate_label, fontsize=FONT_SIZE, verticalalignment='center', horizontalalignment='center')
ax.text(layer
In [108]:
def canvas_plot(qc, height, width):
= plt.subplots(figsize=(width, height))
fig, ax
= qc.N
nwire = 2
display_layers # generalize TODO
draw_wire(ax, nwire)
add_labels(ax, nwire)
# make a dict with wire_labels as keys and zero values
= {f"q{i}": 0 for i in range(nwire)}
gate_maintain
for gate in qc.gates:
if len(gate.targets) == 1 and gate.controls == None:
f"q{gate.targets[0]}"], gate)
add_gate(ax, gate_maintain[f"q{gate.targets[0]}"] += 1
gate_maintain[else:
f"q{gate.targets[0]}"], gate)
multiq_add_gate(ax, gate_maintain[# update all values to multiqubot target
= gate_maintain[f"q{gate.targets[0]}"] + 1
temp = {key:temp for key in gate_maintain.keys()}
gate_maintain
if max(gate_maintain.values()) >= display_layers:
=1)
extend_wire(ax, nwire, display_layers, add_layers
+= 1
display_layers
-WIRE_SEP, nwire*WIRE_SEP)
ax.set_ylim(-LAYER_SEP, (display_layers + 1)*LAYER_SEP)
ax.set_xlim('equal')
ax.set_aspect('off')
ax.axis(200) fig.set_dpi(
In [109]:
def calsize(qc):
= qc.N
n_wires = n_wires * 0.393701 * WIRE_SEP * 3
height = 10
width
return height, width
In [110]:
= QubitCircuit(3)
qc "H", targets=[0])
qc.add_gate("H", targets=[1])
qc.add_gate("X", targets=[2])
qc.add_gate(0].targets, qc.png
qc.gates[
= calsize(qc)
h, w canvas_plot(qc, h, w)
In [111]:
qc
In [112]:
= QubitCircuit(2)
qc "H", targets=[0])
qc.add_gate("H", targets=[1])
qc.add_gate("X", targets=[1])
qc.add_gate(0].targets, qc.png
qc.gates[
= calsize(qc)
h, w canvas_plot(qc, h, w)
In [113]:
= QubitCircuit(2)
qc "H", targets=[0])
qc.add_gate("H", targets=[1])
qc.add_gate("X", targets=[1])
qc.add_gate("CNOT", controls=[0], targets=[1])
qc.add_gate("CNOT", controls=[1], targets=[0])
qc.add_gate("CNOT", controls=[0], targets=[1])
qc.add_gate( qc
In [114]:
= calsize(qc)
h, w canvas_plot(qc, h, w)
In [115]:
"X")
qc.add_1q_gate( canvas_plot(qc, h, w)
In [116]:
= QubitCircuit(3)
qc "H", targets=[0])
qc.add_gate("CNOT", targets=[1], controls=[2])
qc.add_gate("H", targets=[1])
qc.add_gate("X", targets=[1])
qc.add_gate("CNOT", controls=[0], targets=[1])
qc.add_gate("CNOT", controls=[1], targets=[0])
qc.add_gate("CNOT", controls=[0], targets=[2]) qc.add_gate(
In [117]:
qc
In [118]:
= calsize(qc)
h, w canvas_plot(qc, h, w)
In [119]:
= QubitCircuit(3)
qc "H", targets=[0])
qc.add_gate("CNOT", targets=[1], controls=[2])
qc.add_gate("H", targets=[1])
qc.add_gate("Y", targets=[1])
qc.add_gate("X")
qc.add_1q_gate("CNOT", controls=[0], targets=[1])
qc.add_gate("CNOT", controls=[1], targets=[0])
qc.add_gate("CNOT", controls=[0], targets=[2])
qc.add_gate("CNOT", controls=[0], targets=[2])
qc.add_gate( qc
In [120]:
= calsize(qc)
h, w canvas_plot(qc, h, w)