Improved duration display for individual events

Added duration field to event creation dialog
Switched to DuckWidgets for event creation form
Fixed check for events with identical names
Linting
This commit is contained in:
Kumi 2024-04-03 18:22:04 +02:00
parent 22a43d2d1b
commit 35590c0355
Signed by: kumi
GPG key ID: 5D1CE6AF1805ECA2
2 changed files with 275 additions and 124 deletions

View file

@ -1,4 +1,5 @@
import logging import logging
from decimal import Decimal
from collections import OrderedDict from collections import OrderedDict
from PyQt6.QtCore import pyqtSignal, pyqtSlot from PyQt6.QtCore import pyqtSignal, pyqtSlot
from nqrduck.module.module_model import ModuleModel from nqrduck.module.module_model import ModuleModel
@ -6,6 +7,7 @@ from nqrduck_spectrometer.pulsesequence import PulseSequence
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
class PulseProgrammerModel(ModuleModel): class PulseProgrammerModel(ModuleModel):
pulse_parameter_options_changed = pyqtSignal() pulse_parameter_options_changed = pyqtSignal()
events_changed = pyqtSignal() events_changed = pyqtSignal()
@ -16,15 +18,33 @@ class PulseProgrammerModel(ModuleModel):
self.pulse_parameter_options = OrderedDict() self.pulse_parameter_options = OrderedDict()
self.pulse_sequence = PulseSequence("Untitled pulse sequence") self.pulse_sequence = PulseSequence("Untitled pulse sequence")
def add_event(self, event_name): def add_event(self, event_name: str, duration: Decimal = 20):
self.pulse_sequence.events.append(PulseSequence.Event(event_name, "20u")) """Add a new event to the current pulse sequence.
logger.debug("Creating event %s with object id %s", event_name, id(self.pulse_sequence.events[-1]))
Args:
event_name (str): A human-readable name for the event
duration (Decimal): The duration of the event in µs. Defaults to 20.
"""
self.pulse_sequence.events.append(
PulseSequence.Event(event_name, "%.16gu" % duration)
)
logger.debug(
"Creating event %s with object id %s",
event_name,
id(self.pulse_sequence.events[-1]),
)
# Create a default instance of the pulse parameter options and add it to the event # Create a default instance of the pulse parameter options and add it to the event
for name, pulse_parameter_class in self.pulse_parameter_options.items(): for name, pulse_parameter_class in self.pulse_parameter_options.items():
logger.debug("Adding pulse parameter %s to event %s", name, event_name) logger.debug("Adding pulse parameter %s to event %s", name, event_name)
self.pulse_sequence.events[-1].parameters[name] = pulse_parameter_class(name) self.pulse_sequence.events[-1].parameters[name] = pulse_parameter_class(
logger.debug("Created pulse parameter %s with object id %s", name, id(self.pulse_sequence.events[-1].parameters[name])) name
)
logger.debug(
"Created pulse parameter %s with object id %s",
name,
id(self.pulse_sequence.events[-1].parameters[name]),
)
logger.debug(self.pulse_sequence.to_json()) logger.debug(self.pulse_sequence.to_json())
self.events_changed.emit() self.events_changed.emit()

View file

@ -3,12 +3,34 @@ import functools
from collections import OrderedDict from collections import OrderedDict
from pathlib import Path from pathlib import Path
from decimal import Decimal from decimal import Decimal
from PyQt6.QtGui import QIcon from PyQt6.QtGui import QIcon, QValidator
from PyQt6.QtWidgets import QMessageBox, QGroupBox, QFormLayout, QTableWidget, QVBoxLayout, QPushButton, QHBoxLayout, QLabel, QDialog, QLineEdit, QDialogButtonBox, QWidget, QCheckBox, QToolButton, QFileDialog, QSizePolicy from PyQt6.QtWidgets import (
QMessageBox,
QGroupBox,
QFormLayout,
QTableWidget,
QVBoxLayout,
QPushButton,
QHBoxLayout,
QLabel,
QDialog,
QLineEdit,
QDialogButtonBox,
QWidget,
QCheckBox,
QToolButton,
QFileDialog,
QSizePolicy,
)
from PyQt6.QtCore import pyqtSlot, pyqtSignal from PyQt6.QtCore import pyqtSlot, pyqtSignal
from nqrduck.module.module_view import ModuleView from nqrduck.module.module_view import ModuleView
from nqrduck.assets.icons import Logos from nqrduck.assets.icons import Logos
from nqrduck_spectrometer.pulseparameters import BooleanOption, NumericOption, FunctionOption from nqrduck.helpers.duckwidgets import DuckFloatEdit, DuckEdit
from nqrduck_spectrometer.pulseparameters import (
BooleanOption,
NumericOption,
FunctionOption,
)
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
@ -22,21 +44,23 @@ class PulseProgrammerView(ModuleView):
self.setup_variabletables() self.setup_variabletables()
logger.debug("Connecting pulse parameter options changed signal to on_pulse_parameter_options_changed") logger.debug(
self.module.model.pulse_parameter_options_changed.connect(self.on_pulse_parameter_options_changed) "Connecting pulse parameter options changed signal to on_pulse_parameter_options_changed"
)
self.module.model.pulse_parameter_options_changed.connect(
self.on_pulse_parameter_options_changed
)
def setup_variabletables(self) -> None: def setup_variabletables(self) -> None:
"""Setup the table for the variables. """Setup the table for the variables."""
"""
pass pass
def setup_pulsetable(self) -> None: def setup_pulsetable(self) -> None:
"""Setup the table for the pulse sequence. Also add buttons for saving and loading pulse sequences and editing and creation of events """Setup the table for the pulse sequence. Also add buttons for saving and loading pulse sequences and editing and creation of events"""
"""
# Create pulse table # Create pulse table
self.title = QLabel("Pulse Sequence: %s" % self.module.model.pulse_sequence.name) self.title = QLabel(
"Pulse Sequence: %s" % self.module.model.pulse_sequence.name
)
# Make title bold # Make title bold
font = self.title.font() font = self.title.font()
font.setBold(True) font.setBold(True)
@ -44,7 +68,9 @@ class PulseProgrammerView(ModuleView):
# Table setup # Table setup
self.pulse_table = QTableWidget(self) self.pulse_table = QTableWidget(self)
self.pulse_table.setSizeAdjustPolicy(QTableWidget.SizeAdjustPolicy.AdjustToContents) self.pulse_table.setSizeAdjustPolicy(
QTableWidget.SizeAdjustPolicy.AdjustToContents
)
self.pulse_table.setAlternatingRowColors(True) self.pulse_table.setAlternatingRowColors(True)
layout = QVBoxLayout() layout = QVBoxLayout()
button_layout = QHBoxLayout() button_layout = QHBoxLayout()
@ -62,7 +88,9 @@ class PulseProgrammerView(ModuleView):
# Add button for save pulse sequence # Add button for save pulse sequence
self.save_pulse_sequence_button = QPushButton("Save pulse sequence") self.save_pulse_sequence_button = QPushButton("Save pulse sequence")
self.save_pulse_sequence_button.setSizePolicy(QSizePolicy.Policy.Fixed, QSizePolicy.Policy.Fixed) self.save_pulse_sequence_button.setSizePolicy(
QSizePolicy.Policy.Fixed, QSizePolicy.Policy.Fixed
)
# Add the Save Icon to the button # Add the Save Icon to the button
icon = Logos.Save16x16() icon = Logos.Save16x16()
self.save_pulse_sequence_button.setIconSize(icon.availableSizes()[0]) self.save_pulse_sequence_button.setIconSize(icon.availableSizes()[0])
@ -83,7 +111,6 @@ class PulseProgrammerView(ModuleView):
self.module.model.events_changed.connect(self.on_events_changed) self.module.model.events_changed.connect(self.on_events_changed)
self.module.model.pulse_sequence_changed.connect(self.on_pulse_sequence_changed) self.module.model.pulse_sequence_changed.connect(self.on_pulse_sequence_changed)
button_layout.addStretch(1) button_layout.addStretch(1)
layout.addWidget(self.title) layout.addWidget(self.title)
layout.addLayout(button_layout) layout.addLayout(button_layout)
@ -98,19 +125,21 @@ class PulseProgrammerView(ModuleView):
self.on_events_changed() self.on_events_changed()
@pyqtSlot() @pyqtSlot()
def on_pulse_sequence_changed(self) -> None: def on_pulse_sequence_changed(self) -> None:
"""This method is called whenever the pulse sequence changes. It updates the view to reflect the changes. """This method is called whenever the pulse sequence changes. It updates the view to reflect the changes."""
""" logger.debug(
logger.debug("Updating pulse sequence to %s", self.module.model.pulse_sequence.name) "Updating pulse sequence to %s", self.module.model.pulse_sequence.name
)
self.title.setText("Pulse Sequence: %s" % self.module.model.pulse_sequence.name) self.title.setText("Pulse Sequence: %s" % self.module.model.pulse_sequence.name)
@pyqtSlot() @pyqtSlot()
def on_pulse_parameter_options_changed(self) -> None: def on_pulse_parameter_options_changed(self) -> None:
"""This method is called whenever the pulse parameter options change. It updates the view to reflect the changes. """This method is called whenever the pulse parameter options change. It updates the view to reflect the changes."""
""" logger.debug(
logger.debug("Updating pulse parameter options to %s", self.module.model.pulse_parameter_options.keys()) "Updating pulse parameter options to %s",
self.module.model.pulse_parameter_options.keys(),
)
# We set it to the length of the pulse parameter options + 1 because we want to add a row for the parameter option buttons # We set it to the length of the pulse parameter options + 1 because we want to add a row for the parameter option buttons
self.pulse_table.setRowCount(len(self.module.model.pulse_parameter_options) + 1) self.pulse_table.setRowCount(len(self.module.model.pulse_parameter_options) + 1)
# Move the vertical header labels on row down # Move the vertical header labels on row down
@ -120,21 +149,22 @@ class PulseProgrammerView(ModuleView):
@pyqtSlot() @pyqtSlot()
def on_new_event_button_clicked(self) -> None: def on_new_event_button_clicked(self) -> None:
"""This method is called whenever the new event button is clicked. It creates a new event and adds it to the pulse sequence. """This method is called whenever the new event button is clicked. It creates a new event and adds it to the pulse sequence."""
"""
# Create a QDialog for the new event # Create a QDialog for the new event
logger.debug("New event button clicked") logger.debug("New event button clicked")
dialog = AddEventDialog(self) dialog = AddEventDialog(self)
result = dialog.exec() result = dialog.exec()
if result: if result:
event_name = dialog.get_name() event_name = dialog.get_name()
logger.debug("Adding new event with name %s", event_name) duration = dialog.get_duration()
self.module.model.add_event(event_name) logger.debug(
"Adding new event with name %s, duration %g", event_name, duration
)
self.module.model.add_event(event_name, duration)
@pyqtSlot() @pyqtSlot()
def on_events_changed(self) -> None: def on_events_changed(self) -> None:
"""This method is called whenever the events in the pulse sequence change. It updates the view to reflect the changes. """This method is called whenever the events in the pulse sequence change. It updates the view to reflect the changes."""
"""
logger.debug("Updating events to %s", self.module.model.pulse_sequence.events) logger.debug("Updating events to %s", self.module.model.pulse_sequence.events)
# Add label for the event lengths # Add label for the event lengths
@ -145,7 +175,9 @@ class PulseProgrammerView(ModuleView):
for event in self.module.model.pulse_sequence.events: for event in self.module.model.pulse_sequence.events:
logger.debug("Adding event to pulseprogrammer view: %s", event.name) logger.debug("Adding event to pulseprogrammer view: %s", event.name)
# Create a label for the event # Create a label for the event
event_label = QLabel("%s : %s µs" % (event.name, str(event.duration * Decimal(1e6)))) event_label = QLabel(
"%s : %.16g µs" % (event.name, (event.duration * Decimal(1e6)))
)
event_layout.addWidget(event_label) event_layout.addWidget(event_label)
# Delete the old widget and create a new one # Delete the old widget and create a new one
@ -155,33 +187,52 @@ class PulseProgrammerView(ModuleView):
self.layout().addWidget(self.event_widget) self.layout().addWidget(self.event_widget)
self.pulse_table.setColumnCount(len(self.module.model.pulse_sequence.events)) self.pulse_table.setColumnCount(len(self.module.model.pulse_sequence.events))
self.pulse_table.setHorizontalHeaderLabels([event.name for event in self.module.model.pulse_sequence.events]) self.pulse_table.setHorizontalHeaderLabels(
[event.name for event in self.module.model.pulse_sequence.events]
)
self.set_parameter_icons() self.set_parameter_icons()
def set_parameter_icons(self) -> None: def set_parameter_icons(self) -> None:
"""This method sets the icons for the pulse parameter options. """This method sets the icons for the pulse parameter options."""
"""
for column_idx, event in enumerate(self.module.model.pulse_sequence.events): for column_idx, event in enumerate(self.module.model.pulse_sequence.events):
for row_idx, parameter in enumerate(self.module.model.pulse_parameter_options.keys()): for row_idx, parameter in enumerate(
self.module.model.pulse_parameter_options.keys()
):
if row_idx == 0: if row_idx == 0:
event_options_widget = EventOptionsWidget(event) event_options_widget = EventOptionsWidget(event)
# Connect the delete_event signal to the on_delete_event slot # Connect the delete_event signal to the on_delete_event slot
func = functools.partial(self.module.controller.delete_event, event_name=event.name) func = functools.partial(
self.module.controller.delete_event, event_name=event.name
)
event_options_widget.delete_event.connect(func) event_options_widget.delete_event.connect(func)
# Connect the change_event_duration signal to the on_change_event_duration slot # Connect the change_event_duration signal to the on_change_event_duration slot
event_options_widget.change_event_duration.connect(self.module.controller.change_event_duration) event_options_widget.change_event_duration.connect(
self.module.controller.change_event_duration
)
# Connect the change_event_name signal to the on_change_event_name slot # Connect the change_event_name signal to the on_change_event_name slot
event_options_widget.change_event_name.connect(self.module.controller.change_event_name) event_options_widget.change_event_name.connect(
self.module.controller.change_event_name
)
# Connect the move_event_left signal to the on_move_event_left slot # Connect the move_event_left signal to the on_move_event_left slot
event_options_widget.move_event_left.connect(self.module.controller.on_move_event_left) event_options_widget.move_event_left.connect(
self.module.controller.on_move_event_left
)
# Connect the move_event_right signal to the on_move_event_right slot # Connect the move_event_right signal to the on_move_event_right slot
event_options_widget.move_event_right.connect(self.module.controller.on_move_event_right) event_options_widget.move_event_right.connect(
self.module.controller.on_move_event_right
)
self.pulse_table.setCellWidget(row_idx, column_idx, event_options_widget) self.pulse_table.setCellWidget(
self.pulse_table.setRowHeight(row_idx, event_options_widget.layout().sizeHint().height()) row_idx, column_idx, event_options_widget
)
self.pulse_table.setRowHeight(
row_idx, event_options_widget.layout().sizeHint().height()
)
logger.debug("Adding button for event %s and parameter %s", event, parameter) logger.debug(
"Adding button for event %s and parameter %s", event, parameter
)
logger.debug("Parameter object id: %s", id(event.parameters[parameter])) logger.debug("Parameter object id: %s", id(event.parameters[parameter]))
button = QPushButton() button = QPushButton()
icon = event.parameters[parameter].get_pixmap() icon = event.parameters[parameter].get_pixmap()
@ -192,15 +243,21 @@ class PulseProgrammerView(ModuleView):
# We add 1 to the row index because the first row is used for the event options # We add 1 to the row index because the first row is used for the event options
self.pulse_table.setCellWidget(row_idx + 1, column_idx, button) self.pulse_table.setCellWidget(row_idx + 1, column_idx, button)
self.pulse_table.setRowHeight(row_idx + 1, icon.availableSizes()[0].height()) self.pulse_table.setRowHeight(
self.pulse_table.setColumnWidth(column_idx, icon.availableSizes()[0].width()) row_idx + 1, icon.availableSizes()[0].height()
)
self.pulse_table.setColumnWidth(
column_idx, icon.availableSizes()[0].width()
)
# Connect the button to the on_button_clicked slot # Connect the button to the on_button_clicked slot
func = functools.partial(self.on_table_button_clicked, event=event, parameter=parameter) func = functools.partial(
self.on_table_button_clicked, event=event, parameter=parameter
)
button.clicked.connect(func) button.clicked.connect(func)
@pyqtSlot() @pyqtSlot()
def on_table_button_clicked(self, event , parameter) -> None: def on_table_button_clicked(self, event, parameter) -> None:
"""This method is called whenever a button in the pulse table is clicked. It opens a dialog to set the options for the parameter.""" """This method is called whenever a button in the pulse table is clicked. It opens a dialog to set the options for the parameter."""
logger.debug("Button for event %s and parameter %s clicked", event, parameter) logger.debug("Button for event %s and parameter %s clicked", event, parameter)
# Create a QDialog to set the options for the parameter. # Create a QDialog to set the options for the parameter.
@ -209,7 +266,13 @@ class PulseProgrammerView(ModuleView):
if result: if result:
for option, function in dialog.return_functions.items(): for option, function in dialog.return_functions.items():
logger.debug("Setting option %s of parameter %s in event %s to %s", option, parameter, event, function()) logger.debug(
"Setting option %s of parameter %s in event %s to %s",
option,
parameter,
event,
function(),
)
option.set_value(function()) option.set_value(function())
self.set_parameter_icons() self.set_parameter_icons()
@ -232,8 +295,9 @@ class PulseProgrammerView(ModuleView):
if file_name: if file_name:
self.module.controller.load_pulse_sequence(file_name) self.module.controller.load_pulse_sequence(file_name)
class EventOptionsWidget(QWidget): class EventOptionsWidget(QWidget):
""" This class is a widget that can be used to set the options for a pulse parameter. """This class is a widget that can be used to set the options for a pulse parameter.
This widget is then added to the the first row of the according event column in the pulse table. This widget is then added to the the first row of the according event column in the pulse table.
It has a edit button that opens a dialog that allows the user to change the options for the event (name and duration). It has a edit button that opens a dialog that allows the user to change the options for the event (name and duration).
Furthermore it has a delete button that deletes the event from the pulse sequence. Furthermore it has a delete button that deletes the event from the pulse sequence.
@ -299,7 +363,8 @@ class EventOptionsWidget(QWidget):
@pyqtSlot() @pyqtSlot()
def edit_event(self) -> None: def edit_event(self) -> None:
"""This method is called when the edit button is clicked. It opens a dialog that allows the user to change the event name and duration. """This method is called when the edit button is clicked. It opens a dialog that allows the user to change the event name and duration.
If the user clicks ok, the change_event_name and change_event_duration signals are emitted.""" If the user clicks ok, the change_event_name and change_event_duration signals are emitted.
"""
logger.debug("Edit button clicked for event %s", self.event.name) logger.debug("Edit button clicked for event %s", self.event.name)
# Create a QDialog to edit the event # Create a QDialog to edit the event
@ -325,7 +390,9 @@ class EventOptionsWidget(QWidget):
event_form_layout.addRow(duration_layout) event_form_layout.addRow(duration_layout)
layout.addLayout(event_form_layout) layout.addLayout(event_form_layout)
buttons = QDialogButtonBox(QDialogButtonBox.StandardButton.Ok | QDialogButtonBox.StandardButton.Cancel) buttons = QDialogButtonBox(
QDialogButtonBox.StandardButton.Ok | QDialogButtonBox.StandardButton.Cancel
)
buttons.accepted.connect(dialog.accept) buttons.accepted.connect(dialog.accept)
buttons.rejected.connect(dialog.reject) buttons.rejected.connect(dialog.reject)
layout.addWidget(buttons) layout.addWidget(buttons)
@ -336,8 +403,9 @@ class EventOptionsWidget(QWidget):
if name_lineedit.text() != self.event.name: if name_lineedit.text() != self.event.name:
self.change_event_name.emit(self.event.name, name_lineedit.text()) self.change_event_name.emit(self.event.name, name_lineedit.text())
if duration_lineedit.text() != str(self.event.duration): if duration_lineedit.text() != str(self.event.duration):
self.change_event_duration.emit(self.event.name, duration_lineedit.text()) self.change_event_duration.emit(
self.event.name, duration_lineedit.text()
)
@pyqtSlot() @pyqtSlot()
def create_delete_event_dialog(self) -> None: def create_delete_event_dialog(self) -> None:
@ -351,7 +419,9 @@ class EventOptionsWidget(QWidget):
layout = QVBoxLayout() layout = QVBoxLayout()
label = QLabel("Are you sure you want to delete event %s?" % self.event.name) label = QLabel("Are you sure you want to delete event %s?" % self.event.name)
layout.addWidget(label) layout.addWidget(label)
buttons = QDialogButtonBox(QDialogButtonBox.StandardButton.Yes | QDialogButtonBox.StandardButton.No) buttons = QDialogButtonBox(
QDialogButtonBox.StandardButton.Yes | QDialogButtonBox.StandardButton.No
)
buttons.accepted.connect(dialog.accept) buttons.accepted.connect(dialog.accept)
buttons.rejected.connect(dialog.reject) buttons.rejected.connect(dialog.reject)
layout.addWidget(buttons) layout.addWidget(buttons)
@ -373,8 +443,9 @@ class EventOptionsWidget(QWidget):
class OptionsDialog(QDialog): class OptionsDialog(QDialog):
""" This dialog is created whenever the edit button for a pulse option is clicked. """This dialog is created whenever the edit button for a pulse option is clicked.
It allows the user to change the options for the pulse parameter and creates the dialog in accordance to what can be set.""" It allows the user to change the options for the pulse parameter and creates the dialog in accordance to what can be set.
"""
def __init__(self, event, parameter, parent=None): def __init__(self, event, parameter, parent=None):
super().__init__(parent) super().__init__(parent)
@ -444,6 +515,7 @@ class OptionsDialog(QDialog):
self.layout.addWidget(self.buttons) self.layout.addWidget(self.buttons)
class FunctionOptionWidget(QWidget): class FunctionOptionWidget(QWidget):
"""This class is a widget that can be used to set the options for a pulse parameter. """This class is a widget that can be used to set the options for a pulse parameter.
It plots the given function in time and frequency domain. It plots the given function in time and frequency domain.
@ -459,7 +531,9 @@ class FunctionOptionWidget(QWidget):
inner_layout = QHBoxLayout() inner_layout = QHBoxLayout()
for function in function_option.functions: for function in function_option.functions:
button = QPushButton(function.name) button = QPushButton(function.name)
button.clicked.connect(functools.partial(self.on_functionbutton_clicked, function=function)) button.clicked.connect(
functools.partial(self.on_functionbutton_clicked, function=function)
)
inner_layout.addWidget(button) inner_layout.addWidget(button)
layout.addLayout(inner_layout) layout.addLayout(inner_layout)
@ -467,11 +541,13 @@ class FunctionOptionWidget(QWidget):
# Add Advanced settings button # Add Advanced settings button
self.advanced_settings_button = QPushButton("Show Advanced settings") self.advanced_settings_button = QPushButton("Show Advanced settings")
self.advanced_settings_button.clicked.connect(self.on_advanced_settings_button_clicked) self.advanced_settings_button.clicked.connect(
self.on_advanced_settings_button_clicked
)
layout.addWidget(self.advanced_settings_button) layout.addWidget(self.advanced_settings_button)
# Add advanced settings widget # Add advanced settings widget
self.advanced_settings = QGroupBox('Advanced Settings') self.advanced_settings = QGroupBox("Advanced Settings")
self.advanced_settings.setHidden(True) self.advanced_settings.setHidden(True)
self.advanced_settings_layout = QFormLayout() self.advanced_settings_layout = QFormLayout()
self.advanced_settings.setLayout(self.advanced_settings_layout) self.advanced_settings.setLayout(self.advanced_settings_layout)
@ -535,13 +611,14 @@ class FunctionOptionWidget(QWidget):
logger.debug("Invalid expression: %s", self.expr_lineedit.text()) logger.debug("Invalid expression: %s", self.expr_lineedit.text())
self.expr_lineedit.setText(str(self.function_option.value.expr)) self.expr_lineedit.setText(str(self.function_option.value.expr))
# Create message box that tells the user that the expression is invalid # Create message box that tells the user that the expression is invalid
self.create_message_box("Invalid expression", "The expression you entered is invalid. Please enter a valid expression.") self.create_message_box(
"Invalid expression",
"The expression you entered is invalid. Please enter a valid expression.",
)
self.delete_active_function() self.delete_active_function()
self.load_active_function() self.load_active_function()
@pyqtSlot() @pyqtSlot()
def on_advanced_settings_button_clicked(self) -> None: def on_advanced_settings_button_clicked(self) -> None:
"""This function is called when the advanced settings button is clicked. """This function is called when the advanced settings button is clicked.
@ -549,11 +626,10 @@ class FunctionOptionWidget(QWidget):
""" """
if self.advanced_settings.isHidden(): if self.advanced_settings.isHidden():
self.advanced_settings.setHidden(False) self.advanced_settings.setHidden(False)
self.advanced_settings_button.setText('Hide Advanced Settings') self.advanced_settings_button.setText("Hide Advanced Settings")
else: else:
self.advanced_settings.setHidden(True) self.advanced_settings.setHidden(True)
self.advanced_settings_button.setText('Show Advanced Settings') self.advanced_settings_button.setText("Show Advanced Settings")
@pyqtSlot() @pyqtSlot()
def on_functionbutton_clicked(self, function) -> None: def on_functionbutton_clicked(self, function) -> None:
@ -613,7 +689,9 @@ class FunctionOptionWidget(QWidget):
parameter_label = QLabel(parameter.name) parameter_label = QLabel(parameter.name)
parameter_lineedit = QLineEdit(str(parameter.value)) parameter_lineedit = QLineEdit(str(parameter.value))
# Add the parameter_lineedit editingFinished signal to the paramter.set_value slot # Add the parameter_lineedit editingFinished signal to the paramter.set_value slot
parameter_lineedit.editingFinished.connect(lambda: parameter.set_value(parameter_lineedit.text())) parameter_lineedit.editingFinished.connect(
lambda: parameter.set_value(parameter_lineedit.text())
)
# Create a QHBoxLayout # Create a QHBoxLayout
hbox = QHBoxLayout() hbox = QHBoxLayout()
@ -636,7 +714,7 @@ class FunctionOptionWidget(QWidget):
self.end_x_lineedit.setText(str(self.function_option.value.end_x)) self.end_x_lineedit.setText(str(self.function_option.value.end_x))
self.expr_lineedit.setText(str(self.function_option.value.expr)) self.expr_lineedit.setText(str(self.function_option.value.expr))
def create_message_box(self, message : str, information : str) -> None: def create_message_box(self, message: str, information: str) -> None:
"""Creates a message box with the given message and information and shows it. """Creates a message box with the given message and information and shows it.
Args: Args:
@ -652,25 +730,47 @@ class FunctionOptionWidget(QWidget):
class AddEventDialog(QDialog): class AddEventDialog(QDialog):
"""This dialog is created whenever a new event is added to the pulse sequence. It allows the user to enter a name for the event.""" """This dialog is created whenever a new event is added to the pulse sequence. It allows the user to enter a name for the event."""
def __init__(self, parent=None): def __init__(self, parent=None):
super().__init__(parent) super().__init__(parent)
self.setWindowTitle("Add Event") self.setWindowTitle("Add Event")
self.layout = QVBoxLayout(self) self.layout = QFormLayout(self)
self.name_layout = QHBoxLayout()
self.label = QLabel("Enter event name:") self.label = QLabel("Enter event name:")
self.name_input = QLineEdit() self.name_input = DuckEdit()
self.name_input.validator = self.NameInputValidator(self)
self.name_layout.addWidget(self.label)
self.name_layout.addWidget(self.name_input)
self.layout.addRow(self.name_layout)
self.duration_layout = QHBoxLayout()
self.duration_label = QLabel("Duration:")
self.duration_lineedit = DuckFloatEdit(min_value=0)
self.duration_lineedit.setText("20")
self.unit_label = QLabel("µs")
self.duration_layout.addWidget(self.duration_label)
self.duration_layout.addWidget(self.duration_lineedit)
self.duration_layout.addWidget(self.unit_label)
self.layout.addRow(self.duration_layout)
self.buttons = QDialogButtonBox( self.buttons = QDialogButtonBox(
QDialogButtonBox.StandardButton.Ok | QDialogButtonBox.StandardButton.Cancel, QDialogButtonBox.StandardButton.Ok | QDialogButtonBox.StandardButton.Cancel,
self, self,
) )
self.buttons.accepted.connect(self.check_input) self.buttons.accepted.connect(self.check_input)
self.buttons.rejected.connect(self.reject) self.buttons.rejected.connect(self.reject)
self.layout.addWidget(self.label)
self.layout.addWidget(self.name_input)
self.layout.addWidget(self.buttons) self.layout.addWidget(self.buttons)
def get_name(self) -> str: def get_name(self) -> str:
@ -680,20 +780,48 @@ class AddEventDialog(QDialog):
str: The name entered by the user""" str: The name entered by the user"""
return self.name_input.text() return self.name_input.text()
def check_input(self) -> None: def get_duration(self) -> Decimal:
"""Checks if the name entered by the user is valid. If it is, the dialog is accepted. If not, the user is informed of the error. """Returns the duration entered by the user, or a fallback value."
Returns:
Decimal: The duration value provided by the user, or 20"""
return Decimal(self.duration_lineedit.text() or 20)
class NameInputValidator(QValidator):
"""A validator for the name input field.
This is used to validate the input of the QLineEdit widget.
""" """
# Make sure that name is not empty and that event name doesn't already exist. def validate(self, value, position):
if self.name_input.text() == "": """Validates the input value.
self.label.setText("Please enter a name for the event.")
elif self.name_input.text() in self.parent().module.model.pulse_sequence.events: Args:
self.label.setText("Event name already exists. Please enter a different name.") value (str): The input value
else: position (int): The position of the cursor
self.accept()
Returns:
Tuple[QValidator.State, str, int]: The validation state, the fixed value, and the position
"""
if not value:
return (QValidator.State.Intermediate, value, position)
if any(
[
event.name == value
for event in self.parent()
.parent()
.module.model.pulse_sequence.events
]
):
return (QValidator.State.Invalid, value, position)
return (QValidator.State.Acceptable, value, position)
# This class should be refactored in the module view so it can be used by all modules # This class should be refactored in the module view so it can be used by all modules
class QFileManager: class QFileManager:
"""This class provides methods for opening and saving files.""" """This class provides methods for opening and saving files."""
def __init__(self, parent=None): def __init__(self, parent=None):
self.parent = parent self.parent = parent
@ -703,11 +831,13 @@ class QFileManager:
Returns: Returns:
str: The path of the file selected by the user. str: The path of the file selected by the user.
""" """
fileName, _ = QFileDialog.getOpenFileName(self.parent, fileName, _ = QFileDialog.getOpenFileName(
self.parent,
"QFileManager - Open File", "QFileManager - Open File",
"", "",
"Quack Files (*.quack);;All Files (*)", "Quack Files (*.quack);;All Files (*)",
options = QFileDialog.Option.ReadOnly) options=QFileDialog.Option.ReadOnly,
)
if fileName: if fileName:
return fileName return fileName
else: else:
@ -719,16 +849,17 @@ class QFileManager:
Returns: Returns:
str: The path of the file selected by the user. str: The path of the file selected by the user.
""" """
fileName, _ = QFileDialog.getSaveFileName(self.parent, fileName, _ = QFileDialog.getSaveFileName(
self.parent,
"QFileManager - Save File", "QFileManager - Save File",
"", "",
"Quack Files (*.quack);;All Files (*)", "Quack Files (*.quack);;All Files (*)",
options=QFileDialog.Option.DontUseNativeDialog) options=QFileDialog.Option.DontUseNativeDialog,
)
if fileName: if fileName:
# Append the .quack extension if not present # Append the .quack extension if not present
if not fileName.endswith('.quack'): if not fileName.endswith(".quack"):
fileName += '.quack' fileName += ".quack"
return fileName return fileName
else: else:
return None return None