- 
                Notifications
    You must be signed in to change notification settings 
- Fork 15
implemented perform_y #8
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: master
Are you sure you want to change the base?
Changes from all commits
5ec5ba3
              bab9f4b
              cc762eb
              d894d23
              98ad0e3
              f48ac90
              File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change | 
|---|---|---|
|  | @@ -4,6 +4,7 @@ | |
| from sympy.printing.latex import latex | ||
| import drawSvg as draw | ||
| import latextools | ||
| from colorsys import hls_to_rgb | ||
|  | ||
| VERBOSE = False | ||
|  | ||
|  | @@ -138,7 +139,7 @@ def state_text(self, g, time, key, amp=1): | |
| x, y = self.state_xy(key, time) | ||
| g.draw(render_label(sympy.sympify(amp), key), | ||
| x=x+self.w_label/2-self.font*0.2, y=y, scale=self.font/12, center=True, text_anchor='end') | ||
| if abs(float(amp)) < 1e-8: | ||
| if self.get_abs(amp) < 1e-8: | ||
| # Draw red X over it | ||
| ys = self.font/2*1.4 | ||
| xs = self.w_label/2*0.7 | ||
|  | @@ -166,7 +167,7 @@ def straight_arrow(self, g, color, *xy_list, width=1): | |
| marker_start=self.make_arrow(color))) | ||
|  | ||
| def gate_arrow(self, g, time1, key1, key2, amp=1): | ||
| w = float(abs(amp)) | ||
| w = max(float(abs(amp)),.1) | ||
| x1, y1 = self.state_xy(key1, time1) | ||
| x2, y2 = self.state_xy(key2, time1+1) | ||
| x1 += self.arrow_off | ||
|  | @@ -175,47 +176,57 @@ def gate_arrow(self, g, time1, key1, key2, amp=1): | |
| xx2 = x2 - self.w_label/2 + self.arrow_off | ||
| yy1 = y1 + (y2-y1)*(xx1-x1)/(x2-x1) | ||
| yy2 = y2 - (y2-y1)*(x2-xx2)/(x2-x1) | ||
| color = '#26f' | ||
| if abs(float(amp) - abs(float(amp))) >= 1e-8: | ||
| color = '#e70' | ||
| # map the amplitude to a "wheel" of colors | ||
| color = self.wheel_color(amp) | ||
| self.straight_arrow(g, color, xx1, yy1, xx2, yy2, width=w) | ||
|  | ||
| def wheel_color(self, amp): | ||
| """ Maps the amplitude to a color wheel. """ | ||
| offset = 7*sympy.pi/6 # offset to have blue for angle zero | ||
| angle = sympy.arg(amp)+ offset | ||
| angle = angle % (2*sympy.pi) | ||
| angle = (angle / (2*sympy.pi)).evalf(2) | ||
| rgb = hls_to_rgb(angle,0.6,1) | ||
| rgb = tuple(int(x*255) for x in rgb) | ||
| return '#%02x%02x%02x' % (rgb) | ||
| def get_abs(self, amp): | ||
| """ | ||
| Check if the amplitude is complex. If so, return the magnitude. | ||
| """ | ||
| if isinstance(amp, sympy.Expr): | ||
| if amp.is_complex: | ||
| return abs(amp) | ||
| return abs(float(amp)) | ||
|  | ||
| def get_amp_as_value(self, amp, n, digits=2): | ||
| """ | ||
| Check if the amplitude is a sympy element. | ||
| If so, and has more than n ops return the value in digits | ||
| """ | ||
| if hasattr(amp, "count_ops"): | ||
| if amp.count_ops() > n: | ||
| amp = amp.evalf(digits) | ||
| return amp | ||
|  | ||
| def draw_states(self): | ||
| t = len(self.state_sequence)-1 | ||
| for key, amp in self.state_sequence[-1].items(): | ||
| amp = self.get_amp_as_value(amp,n=4) | ||
| #amp = 0 if abs(amp)<1e-8 else amp | ||
| self.state_text(self.d, t, key, amp=amp) | ||
|  | ||
| def add_states(self, new_state): | ||
| for key,amp in new_state.items(): | ||
| new_state[key]= sympy.simplify(amp) | ||
| self.state_sequence.append(new_state) | ||
| self.draw_states() | ||
| clean_state = { | ||
| key: amp | ||
| for key, amp in new_state.items() | ||
| if abs(float(amp)) >= 1e-8 | ||
| if self.get_abs(amp) >= 1e-8 | ||
| } | ||
| self.state_sequence[-1] = clean_state | ||
|  | ||
| def perform_h(self, q_i, *, pre_latex=f'', name='H'): | ||
| new_state = {} | ||
| t = len(self.state_sequence)-1 | ||
| for key, amp in self.state_sequence[-1].items(): | ||
| is_one = key[q_i] == '1' | ||
| digits = list(key) | ||
| digits[q_i] = '0' | ||
| zero = ''.join(digits) | ||
| digits[q_i] = '1' | ||
| one = ''.join(digits) | ||
| zero_amp = 1/sympy.sqrt(2) | ||
| one_amp = -zero_amp if is_one else zero_amp | ||
| self.gate_arrow(self.d, t, key, zero, amp=zero_amp) | ||
| self.gate_arrow(self.d, t, key, one, amp=one_amp) | ||
| if zero not in new_state: new_state[zero] = 0 | ||
| if one not in new_state: new_state[one] = 0 | ||
| new_state[zero] += amp*zero_amp | ||
| new_state[one] += amp*one_amp | ||
| self.transition_text(self.d, t, f'{pre_latex}{name}_{q_i}') | ||
| self.add_states(new_state) | ||
|  | ||
| def perform_cnot(self, qi1, qi2, *, pre_latex=f'', name='CNOT'): | ||
| new_state = {} | ||
| t = len(self.state_sequence)-1 | ||
|  | @@ -226,35 +237,76 @@ def perform_cnot(self, qi1, qi2, *, pre_latex=f'', name='CNOT'): | |
| if is_one: | ||
| digits[qi2] = '01'[not is_targ_one] | ||
| new_key = ''.join(digits) | ||
| self.gate_arrow(self.d, t, key, new_key, amp=1) | ||
| self.gate_arrow(self.d, t, key, new_key, amp=amp) | ||
| if new_key not in new_state: new_state[new_key] = 0 | ||
| new_state[new_key] += amp | ||
| self.transition_text(self.d, t, f'{pre_latex}{name}_{{{qi1}{qi2}}}') | ||
| self.add_states(new_state) | ||
|  | ||
| def perform_z(self, q_i, *, pre_latex=f'', name='Z'): | ||
| new_state = {} | ||
| t = len(self.state_sequence)-1 | ||
| for key, amp in self.state_sequence[-1].items(): | ||
| is_one = key[q_i] == '1' | ||
| new_amp = -amp if is_one else amp | ||
| self.gate_arrow(self.d, t, key, key, amp=new_amp/amp) | ||
| if key not in new_state: new_state[key] = 0 | ||
| new_state[key] += new_amp | ||
| self.transition_text(self.d, t, f'{pre_latex}{name}_{{{q_i}}}') | ||
| self.add_states(new_state) | ||
|  | ||
| def perform_x(self, q_i, *, pre_latex=f'', name='X'): | ||
| X_gate = [[0, 1], [1, 0]] | ||
| self.perform_single_gate( q_i, pre_latex, name, X_gate) | ||
|  | ||
| def perform_y(self, q_i, *, pre_latex=f'', name='Y'): | ||
| Y_gate = [[0, -sympy.I], [sympy.I, 0]] | ||
| self.perform_single_gate( q_i, pre_latex, name, Y_gate) | ||
|  | ||
| def perform_z(self, q_i, *, pre_latex=f'', name='Z'): | ||
| Z_gate = [[1, 0], [0, -1]] | ||
| self.perform_single_gate( q_i, pre_latex, name, Z_gate) | ||
|  | ||
| def perform_h(self, q_i, *, pre_latex=f'', name='H'): | ||
| sqrt2 = sympy.sqrt(2) | ||
| H_gate = [[1/sqrt2, 1/sqrt2], | ||
| [1/sqrt2, -1/sqrt2]] | ||
| self.perform_single_gate( q_i, pre_latex, name, H_gate) | ||
|  | ||
| def perform_rx(self, q_i, half_turns, *, pre_latex=f'', name='Rx'): | ||
| theta = sympy.pi*half_turns | ||
| Rx_gate = [[sympy.cos(theta/2), -sympy.I*sympy.sin(theta/2)], | ||
| [-sympy.I*sympy.sin(theta/2), sympy.cos(theta/2)]] | ||
| self.perform_single_gate( q_i, pre_latex, name, Rx_gate) | ||
|  | ||
| def perform_ry(self, q_i, half_turns, *, pre_latex=f'', name='Ry'): | ||
| theta = sympy.pi*half_turns | ||
| Ry_gate = [[sympy.cos(theta/2), -sympy.sin(theta/2)], | ||
| [sympy.sin(theta/2), sympy.cos(theta/2)]] | ||
| self.perform_single_gate( q_i, pre_latex, name, Ry_gate) | ||
|  | ||
| def perform_rz(self, q_i, half_turns, *, pre_latex=f'', name='Rz'): | ||
| theta = sympy.pi*half_turns | ||
| Rz_gate = [[ sympy.exp(-sympy.I*theta/2), 0], | ||
| [0, sympy.exp(sympy.I*theta/2)]] | ||
| self.perform_single_gate( q_i, pre_latex, name, Rz_gate) | ||
|  | ||
| def perform_single_gate(self, q_i, pre_latex, name, gate_matrix): | ||
| new_state = {} | ||
| t = len(self.state_sequence)-1 | ||
| for key, amp in self.state_sequence[-1].items(): | ||
| is_one = key[q_i] == '1' | ||
| digits = list(key) | ||
| digits[q_i] = '01'[not is_one] | ||
| new_key = ''.join(digits) | ||
| self.gate_arrow(self.d, t, key, new_key, amp=1) | ||
| if new_key not in new_state: | ||
| new_state[new_key] = 0 | ||
| new_state[new_key] += amp | ||
| self.transition_text(self.d, t, f'{pre_latex}{name}_{{{q_i}}}') | ||
| digits[q_i] = '0' | ||
| zero = ''.join(digits) | ||
| digits[q_i] = '1' | ||
| one = ''.join(digits) | ||
| if is_one: | ||
| zero_amp = gate_matrix[0][1] | ||
| one_amp = gate_matrix[1][1] | ||
| else: | ||
| zero_amp = gate_matrix[0][0] | ||
| one_amp = gate_matrix[1][0] | ||
|  | ||
| if zero_amp != 0: | ||
| self.gate_arrow(self.d, t, key, zero, amp=zero_amp*amp) | ||
| if zero not in new_state: new_state[zero] = 0 | ||
| new_state[zero] += zero_amp*amp | ||
| if one_amp != 0: | ||
| self.gate_arrow(self.d, t, key, one, amp=one_amp*amp) | ||
| if one not in new_state: new_state[one] = 0 | ||
| new_state[one] += one_amp*amp | ||
|  | ||
| self.transition_text(self.d, t, f'{pre_latex}{name}_{q_i}') | ||
| self.add_states(new_state) | ||
| There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Nice. If you intend to add many more gates, can you refactor the perform_* methods to avoid the code copy? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Will do. Shall I make a perform_gate class? Or do you have something better in mind? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It could be as simple as a dictionary with symbolic gate unitaries (at least for single-qubit gates:  | ||
|  | ||
|  | ||
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No need to do this. Just use pi from
import math.