import numpy as np, scipy.optimize import traceback from PyQt5.QtWidgets import * from PyQt5.QtGui import * from PyQt5.QtCore import * import phys, file # global GUI-related shit class GUI: table = None table_rows = list() monospace = QFont("monospace") bigfont = QFont("sans-serif", pointSize=10, weight=1000) class Pens: axes = QPen(Qt.gray) ellipse = QPen(Qt.black) ellipse.setWidth(2) axis_linear = QPen(QColor(201, 141, 0)) axis_linear.setWidth(2) axis_linear.setStyle(Qt.DashDotLine) axis_fast = QPen(QColor(51, 87, 123)) axis_fast.setWidth(2) axis_fast.setStyle(Qt.DashDotLine) axis_slow = QPen(QColor(55, 123, 51)) axis_slow.setWidth(2) axis_slow.setStyle(Qt.DashDotLine) alpha = QPen(Qt.red) theta = QPen(Qt.blue) radii = QPen(Qt.black) radii.setStyle(Qt.DashDotLine) from ui_widgets import * class OptBox(QVBoxLayout): def __init__(self, row, pol, rownum): super().__init__() self.rownum = rownum # Name self.name = MeinGroßLabel("Element %d" % rownum) self.name.setContextMenuPolicy(Qt.CustomContextMenu) self.name.customContextMenuRequested.connect(lambda pos: OptBox.open_menu(self, pos)) self.addWidget(self.name) # Type / Enable box box = QHBoxLayout() self.addLayout(box) # Type combo self.type = QComboBox() self.type.addItem("Linear", "linear",) self.type.addItem("Quarterwave plate", "quarterwave") self.type.setCurrentIndex(0 if pol.type == "linear" else 1) self.type.currentIndexChanged.connect(lambda : TableRow.type_change(row)) box.addWidget(self.type) # Enable checkbox self.enable = QCheckBox("Enable") self.enable.setChecked(True) self.enable.stateChanged.connect(update) box.addWidget(self.enable) # Delta / Reference layout box = QHBoxLayout() self.addLayout(box) # Angle reference self.ref = QComboBox() self.ref.addItem("Absolute", False) for i, _ in enumerate(system.elements): if i >= rownum - 1: continue self.ref.addItem("Relative to #%d" % (i + 1), i) if row.ref is not False and i == row.ref: self.ref.setCurrentIndex(row.ref + 1) self.ref.currentIndexChanged.connect(update) box.addWidget(self.ref) # Delta angle box.addWidget(QLabel("Delta")) self.delta = QLineEdit("0") self.delta.textChanged.connect(lambda: TableRow.angle_change(row)) box.addWidget(self.delta) box.addWidget(QLabel("°")) # Transmittance layout box = QHBoxLayout() self.addLayout(box) box.addWidget(QLabel("T")) self.t1 = QLineEdit("1") self.t1.textChanged.connect(lambda: TableRow.t_change(row)) box.addWidget(self.t1) self.t2 = QLineEdit("0") self.t2.textChanged.connect(lambda: TableRow.t_change(row)) box.addWidget(self.t2) # Angle self.angle = AngleSlider("Angle") self.angle.setValue(pol.angle * 180 / np.pi) self.angle.on_change = lambda: TableRow.angle_change(row) self.addLayout(self.angle) self.addItem(ExpandingSpacer()) def open_menu(self, pos): menu = QMenu() menu.addAction("Insert before", lambda: half_assed_element_creation(self.rownum - 1)) menu.addAction("Insert after", lambda: half_assed_element_creation(self.rownum)) menu.addAction("Delete", lambda: half_assed_element_deletion(self.rownum - 1)) menu.exec_(self.name.mapToGlobal(pos)) class TableRow: def __init__(self, pol, grid, rownum): self.pol = pol self.grid = grid self.rownum = rownum self.angle = pol.angle self.delta = pol.delta self.ref = pol.ref self.optbox = OptBox(self, pol, rownum) grid.addLayout(self.optbox, rownum, 1) self.ellipse = EllipseWidget(pol) grid.addWidget(self.ellipse, rownum, 2) self.info = QVBoxLayout() self.info_angle = MeinLabel() self.info_angle.setAlignment(Qt.AlignLeft) self.info.addWidget(self.info_angle) self.info.addWidget(QLabel("Polarization state")) self.info_jones = MeinLabel() self.info_jones.setAlignment(Qt.AlignLeft) self.info.addWidget(self.info_jones) self.info.addItem(ExpandingSpacer()) grid.addLayout(self.info, rownum, 3) @staticmethod def angle_change(row): row.angle = row.optbox.angle.angle / 180 * np.pi try: row.delta = float(row.optbox.delta.text()) / 180 * np.pi except ValueError: pass update() @staticmethod def type_change(row): row.pol.set_type(row.optbox.type.currentData()) update() @staticmethod def t_change(row): try: row.pol.t1 = float(row.optbox.t1.text()) row.pol.t2 = float(row.optbox.t2.text()) except ValueError: pass update() def populate_table(): GUI.table = QGridLayout() GUI.table.setColumnStretch(1, 1) GUI.table.setColumnStretch(2, 2) GUI.table.setColumnStretch(3, 5) while GUI.opt_operand.count() > 0: # FIXME GUI.opt_operand.removeItem(0) GUI.table_rows = list() for i, pol in enumerate(system.elements): GUI.table_rows.append(TableRow(pol, GUI.table, i + 1)) GUI.opt_operand.addItem("Element #%d" % (i + 1), i) def update(): system.ignore = [False] * len(system.elements) for i, pol in enumerate(system.elements): row = GUI.table_rows[i] pol.angle = row.angle pol.delta = row.delta pol.ref = row.optbox.ref.currentData() if pol.ref is not False: pol.angle += system.elements[pol.ref].angle if not row.optbox.enable.isChecked(): system.ignore[i] = True system.recalculate() I = 1 for i, pol in enumerate(system.elements): row = GUI.table_rows[i] # update all the diagrams row.ellipse.state = system.states[i] row.ellipse.ellipse = system.ellipses[i] row.ellipse.is_used = not system.ignore[i] row.ellipse.repaint() text = "%s at %f°" \ % (pol.type, (pol.angle + pol.delta) * 180 / np.pi) row.info_angle.setText(text) state = system.states[i] ellipse = system.ellipses[i] if state is None: I = 1 row.info_jones.setText("Unpolarized") else: I = np.abs(state[0] * state[0].conjugate() \ + state[1] * state[1].conjugate()) if state is None: continue A, B = state text = "I = %f\n" % I text += "%f ∠ %+f°\n%f ∠ %+f°\n" % \ (np.abs(A), 180 / np.pi * np.angle(A), \ np.abs(B), 180 / np.pi * np.angle(B)) text += "α = %f°\nφ = %+f°" % \ (180 / np.pi * ellipse.alpha, 180 / np.pi * ellipse.theta) text = text.replace("-", "- ").replace("+", "+ ") row.info_jones.setText(text) GUI.widok.intensity = I GUI.widok.repaint() #if GUI.auto_optimize.isChecked(): # optimize() #TODO later #class AddElementWindow(QMainWindow): # dialog = None # # def __init__(self): # super().__init__() # # chuj = QLabel("adolf") # self.setCentralWidget(chuj) # self.show() # # # def __del__(self): # AddElementWindow.dialog = None # # def open(): # if not AddElementWindow.dialog: # AddElementWindow.dialog = AddElementWindow() def half_assed_element_creation(index=None): if index is None: system.elements.append(phys.Polarizer("linear")) else: system.elements.insert(index, phys.Polarizer("linear")) populate_table() GUI.table_frame = QFrame() GUI.table_frame.setLayout(GUI.table) GUI.scroll.setWidget(GUI.table_frame) update() def half_assed_clear(): system.elements = list() populate_table() GUI.table_frame = QFrame() GUI.table_frame.setLayout(GUI.table) GUI.scroll.setWidget(GUI.table_frame) update() def half_assed_element_deletion(index): del(system.elements[index]) populate_table() GUI.table_frame = QFrame() GUI.table_frame.setLayout(GUI.table) GUI.scroll.setWidget(GUI.table_frame) update() last_save_path = None file_filter = "Wery Omportant Zejta (*.woz)" def do_save(win, reuse_old): global system, last_save_path if reuse_old and last_save_path: path = last_save_path else: path, _ = QFileDialog.getSaveFileName(win, filter=file_filter) if path == "": return try: file.save_system(path, system) except: traceback.print_exc() last_save_path = path def do_open(win): global system path, _ = QFileDialog.getOpenFileName(win, filter=file_filter) if path == "": return try: system = file.open_system(path) except: traceback.print_exc() populate_table() GUI.table_frame = QFrame() GUI.table_frame.setLayout(GUI.table) GUI.scroll.setWidget(GUI.table_frame) update() def setup_menubar(win): menu = win.menuBar() # File open = QAction("&Open a system", win) open.setShortcut("Ctrl+O") open.triggered.connect(lambda: do_open(win)) save = QAction("&Save a system", win) save.setShortcut("Ctrl+S") save.triggered.connect(lambda: do_save(win, True)) save_as = QAction("&Save a system as...", win) save_as.triggered.connect(lambda: do_save(win, False)) close = QAction("Exit", win) close.setShortcut("Ctrl+Q") close.triggered.connect(exit) file = menu.addMenu("&File") file.addAction(open) file.addAction(save) file.addAction(save_as) file.addAction(close) # System add = QAction("&Add a new element", win) add.setShortcut("Ctrl+N") add.triggered.connect(half_assed_element_creation) clear = QAction("&Remove all elements", win) clear.triggered.connect(half_assed_clear) system = menu.addMenu("&System") system.addAction(add) system.addAction(clear) win.statusBar() # FIXME: refactor def optimize(which): if len(system.elements) == 0: return op_idx = GUI.opt_operand.currentData() op = system.elements[op_idx] def I(theta): op.angle = theta if op.ref is not False: op.angle += system.elements[op.ref].angle state = None for i, pol in enumerate(system.elements): if i >= len(system.ignore) or not system.ignore[i]: state = pol.mul(state) if state is None: return 1 else: return float(np.abs(state[0] * state[0].conjugate() \ + state[1] * state[1].conjugate())) if which == "min": f = I else: f = lambda theta: -I(theta) opt = scipy.optimize.minimize_scalar(f, bounds=[0, np.pi], method="bounded") GUI.table_rows[op_idx].optbox.angle.edit.setText("%g" % round(opt.x * 180 / np.pi, 3)) def setup(win, system_): global system system = system_ # Needless to say, I'm getting real tired of this whole Layout/Widget # clusterfuck in Qt root = QHBoxLayout() root_fucking_random_container = QWidget() root_fucking_random_container.setLayout(root) win.setCentralWidget(root_fucking_random_container) rhs = QVBoxLayout() Widocques.image = QImage("jones.jpg") GUI.widok = Widocques() rhs.addWidget(GUI.widok) optbox = QHBoxLayout() rhs.addLayout(optbox) GUI.opt_operand = QComboBox() optbox.addWidget(GUI.opt_operand) button = QPushButton("Find a minimum") button.pressed.connect(lambda: optimize("min")) optbox.addWidget(button) button = QPushButton("Find a maximum") button.pressed.connect(lambda: optimize("max")) optbox.addWidget(button) GUI.scroll = QScrollArea() populate_table() GUI.table_frame = QFrame() GUI.table_frame.setLayout(GUI.table) GUI.scroll.setWidget(GUI.table_frame) root.addWidget(GUI.scroll) root.addLayout(rhs) setup_menubar(win) update() win.show()