Skip to content
Snippets Groups Projects
signal_generator_input.py 8.29 KiB
Newer Older
  • Learn to ignore specific revisions
  • # -*- coding: utf-8 -*-
    
    from qtpy.QtWidgets import (
        QFileDialog,
        QGridLayout,
        QLabel,
        QLineEdit,
        QPushButton,
        QSpinBox,
    )
    
    
    from b_asic.signal_generator import (
        Constant,
    
        FromFile,
    
        Gaussian,
        Impulse,
        SignalGenerator,
        Sinusoid,
        Step,
        Uniform,
        ZeroPad,
    )
    
    
    class SignalGeneratorInput(QGridLayout):
        """Abstract class for graphically configuring and generating signal generators."""
    
        def __init__(self, logger, *args, **kwargs):
            super().__init__(*args, **kwargs)
            self._logger = logger
    
        def get_generator(self) -> SignalGenerator:
            """Return the SignalGenerator based on the graphical input."""
            raise NotImplementedError
    
    
        def _parse_number(self, string, _type, name, default):
            string = string.strip()
            try:
                if not string:
                    return default
                return _type(string)
            except ValueError:
                self._logger.warning(
                    f"Cannot parse {name}: {string} not a {_type.__name__}, setting to"
                    f" {default}"
                )
                return default
    
    
    
    class DelayInput(SignalGeneratorInput):
        """
        Abstract class for graphically configuring and generating signal generators that
        have a single delay parameter.
        """
    
        def __init__(self, *args, **kwargs):
            super().__init__(*args, **kwargs)
            self.delay_label = QLabel("Delay")
            self.addWidget(self.delay_label, 0, 0)
            self.delay_spin_box = QSpinBox()
            self.delay_spin_box.setRange(0, 2147483647)
            self.addWidget(self.delay_spin_box, 0, 1)
    
        def get_generator(self) -> SignalGenerator:
            raise NotImplementedError
    
    
    class ImpulseInput(DelayInput):
        """
        Class for graphically configuring and generating a
        :class:`~b_asic.signal_generators.Impulse` signal generator.
        """
    
        def get_generator(self) -> SignalGenerator:
            return Impulse(self.delay_spin_box.value())
    
    
    class StepInput(DelayInput):
        """
        Class for graphically configuring and generating a
        :class:`~b_asic.signal_generators.Step` signal generator.
        """
    
        def get_generator(self) -> SignalGenerator:
            return Step(self.delay_spin_box.value())
    
    
    class ZeroPadInput(SignalGeneratorInput):
        """
        Class for graphically configuring and generating a
        :class:`~b_asic.signal_generators.ZeroPad` signal generator.
        """
    
        def __init__(self, *args, **kwargs):
            super().__init__(*args, **kwargs)
            self.input_label = QLabel("Input")
            self.addWidget(self.input_label, 0, 0)
            self.input_sequence = QLineEdit()
    
            self.input_sequence.setPlaceholderText("0.1, -0.2, 0.7")
    
            self.addWidget(self.input_sequence, 0, 1)
    
        def get_generator(self) -> SignalGenerator:
            input_values = []
            for val in self.input_sequence.text().split(","):
                val = val.strip()
                try:
                    if not val:
                        val = 0
                    val = complex(val)
                except ValueError:
                    self._logger.warning(f"Skipping value: {val}, not a digit.")
                    continue
                input_values.append(val)
            return ZeroPad(input_values)
    
    
    
    class FromFileInput(SignalGeneratorInput):
        """
        Class for graphically configuring and generating a
        :class:`~b_asic.signal_generators.FromFile` signal generator.
        """
    
        def __init__(self, *args, **kwargs):
            super().__init__(*args, **kwargs)
            self.file_label = QLabel("Browse")
            self.addWidget(self.file_label, 0, 0)
            self.file_browser = QPushButton("No file selected")
            self.file_browser.clicked.connect(
                lambda i: self.get_input_file(i, self.file_browser)
            )
            self.addWidget(self.file_browser, 0, 1)
    
        def get_generator(self) -> SignalGenerator:
            return FromFile(self.file_browser.text())
    
        def get_input_file(self, i, file_browser):
            module, accepted = QFileDialog().getOpenFileName()
            file_browser.setText(module)
            return
    
    
    
    class SinusoidInput(SignalGeneratorInput):
        """
        Class for graphically configuring and generating a
        :class:`~b_asic.signal_generators.Sinusoid` signal generator.
        """
    
        def __init__(self, *args, **kwargs):
            super().__init__(*args, **kwargs)
            self.frequency_label = QLabel("Frequency")
            self.addWidget(self.frequency_label, 0, 0)
            self.frequency_input = QLineEdit()
    
            self.frequency_input.setText("0.1")
    
            self.addWidget(self.frequency_input, 0, 1)
    
            self.phase_label = QLabel("Phase")
            self.addWidget(self.phase_label, 1, 0)
            self.phase_input = QLineEdit()
    
            self.phase_input.setText("0.0")
    
            self.addWidget(self.phase_input, 1, 1)
    
        def get_generator(self) -> SignalGenerator:
    
            frequency = self._parse_number(
                self.frequency_input.text(), float, "Frequency", 0.1
            )
            phase = self._parse_number(self.phase_input.text(), float, "Phase", 0.0)
    
            return Sinusoid(frequency, phase)
    
    
    class GaussianInput(SignalGeneratorInput):
        """
        Class for graphically configuring and generating a
        :class:`~b_asic.signal_generators.Gaussian` signal generator.
        """
    
        def __init__(self, *args, **kwargs):
            super().__init__(*args, **kwargs)
            self.scale_label = QLabel("Standard deviation")
            self.addWidget(self.scale_label, 0, 0)
            self.scale_input = QLineEdit()
            self.scale_input.setText("1.0")
            self.addWidget(self.scale_input, 0, 1)
    
            self.loc_label = QLabel("Average value")
            self.addWidget(self.loc_label, 1, 0)
            self.loc_input = QLineEdit()
            self.loc_input.setText("0.0")
            self.addWidget(self.loc_input, 1, 1)
    
            self.seed_label = QLabel("Seed")
            self.addWidget(self.seed_label, 2, 0)
            self.seed_spin_box = QSpinBox()
            self.seed_spin_box.setRange(0, 2147483647)
            self.addWidget(self.seed_spin_box, 2, 1)
    
        def get_generator(self) -> SignalGenerator:
    
            scale = self._parse_number(
                self.scale_input.text(), float, "Standard deviation", 1.0
            )
            loc = self._parse_number(self.loc_input.text(), float, "Average value", 0.0)
    
            return Gaussian(self.seed_spin_box.value(), loc, scale)
    
    
    class UniformInput(SignalGeneratorInput):
        """
        Class for graphically configuring and generating a
        :class:`~b_asic.signal_generators.Uniform` signal generator.
        """
    
        def __init__(self, *args, **kwargs):
            super().__init__(*args, **kwargs)
            self.low_label = QLabel("Lower bound")
            self.addWidget(self.low_label, 0, 0)
            self.low_input = QLineEdit()
            self.low_input.setText("-1.0")
            self.addWidget(self.low_input, 0, 1)
    
            self.high_label = QLabel("Upper bound")
            self.addWidget(self.high_label, 1, 0)
            self.high_input = QLineEdit()
            self.high_input.setText("1.0")
            self.addWidget(self.high_input, 1, 1)
    
            self.seed_label = QLabel("Seed")
            self.addWidget(self.seed_label, 2, 0)
            self.seed_spin_box = QSpinBox()
            self.seed_spin_box.setRange(0, 2147483647)
            self.addWidget(self.seed_spin_box, 2, 1)
    
        def get_generator(self) -> SignalGenerator:
    
            low = self._parse_number(self.low_input.text(), float, "Lower bound", -1.0)
            high = self._parse_number(self.high_input.text(), float, "Upper bound", 1.0)
    
            return Uniform(self.seed_spin_box.value(), low, high)
    
    
    class ConstantInput(SignalGeneratorInput):
        """
        Class for graphically configuring and generating a
        :class:`~b_asic.signal_generators.Constant` signal generator.
        """
    
        def __init__(self, *args, **kwargs):
            super().__init__(*args, **kwargs)
            self.constant_label = QLabel("Constant")
            self.addWidget(self.constant_label, 0, 0)
            self.constant_input = QLineEdit()
            self.constant_input.setText("1.0")
            self.addWidget(self.constant_input, 0, 1)
    
        def get_generator(self) -> SignalGenerator:
    
            constant = self._parse_number(
                self.constant_input.text(), complex, "Constant", 1.0
            )
    
            return Constant(constant)
    
    
    _GENERATOR_MAPPING = {
        "Constant": ConstantInput,
        "Gaussian": GaussianInput,
        "Impulse": ImpulseInput,
        "Sinusoid": SinusoidInput,
        "Step": StepInput,
        "Uniform": UniformInput,
        "ZeroPad": ZeroPadInput,
    
        "FromFile": FromFileInput,