Module xelo2.gui.modal

Expand source code
from logging import getLogger
from pathlib import Path
from PyQt5.QtWidgets import (
    QAction,
    QComboBox,
    QDialog,
    QDialogButtonBox,
    QDoubleSpinBox,
    QFileDialog,
    QFormLayout,
    QGridLayout,
    QHBoxLayout,
    QLabel,
    QLineEdit,
    QMenu,
    QPushButton,
    QSizePolicy,
    QTableWidget,
    QTableWidgetItem,
    QVBoxLayout,
    )
from PyQt5.QtCore import Qt

from functools import partial
from numpy import issubdtype, floating, integer

from .utils import _protocol_name
from ..api.filetype import parse_filetype
from ..api.frontend import list_experimenters
from ..database.tables import LEVELS, lookup_allowed_values
from ..io.ephys import read_info_from_ephys
from ..io.utils import localize_blackrock
from ..io.events import read_events_from_ephys

lg = getLogger(__name__)


class NewFile(QDialog):

    def __init__(self, parent, file_obj=None, level_obj=None, filename=None):
        super().__init__(parent)
        self.setWindowModality(Qt.WindowModal)

        self.level = QComboBox()
        self.level.addItems([level[:-1].capitalize() for level in LEVELS])

        if level_obj is not None:
            self.level.setEnabled(False)  # do not allow changing level here
            self.level.setCurrentText(level_obj.t.capitalize())

        self.filepath = QLineEdit()
        self.filepath.setFixedWidth(800)
        self.filepath.editingFinished.connect(self.set_filetype)
        browse = QPushButton('Browse ...')
        browse.clicked.connect(self.browse)
        self.format = QComboBox()
        self.format.addItems(['Unknown', ] + lookup_allowed_values(parent.db['db'], 'files', 'format'))

        bbox = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
        bbox.accepted.connect(self.accept)
        bbox.rejected.connect(self.reject)

        layout_file = QHBoxLayout()
        layout_file.addWidget(self.filepath)
        layout_file.addWidget(browse)

        layout = QVBoxLayout()
        layout.addWidget(self.level)
        layout.addLayout(layout_file)
        layout.addWidget(self.format)
        layout.addWidget(bbox)

        self.setLayout(layout)

        if file_obj is not None:
            # self.level.setCurrentText(file_obj)
            self.filepath.setText(str(file_obj.path))
            self.format.setCurrentText(file_obj.format)

        if filename is not None:
            self.filepath.setText(filename)
            self.set_filetype()
            self.filepath.setEnabled(False)
            self.set_filetype(filename)

    def browse(self):
        filename, _ = QFileDialog.getOpenFileName(self, 'Select File')

        if filename:
            self.filepath.setText(filename)
            self.set_filetype(filename)

    def set_filetype(self, filename=None):
        if filename is None:
            filename = Path(self.filepath.text()).resolve()

        try:
            filetype = parse_filetype(filename)

        except ValueError as err:
            lg.debug(err)
            print(err)

        else:
            self.format.setCurrentText(filetype)


class CalculateOffset(QDialog):
    def __init__(self, parent, file_fixed, file_moving):
        super().__init__(parent)

        t0 = localize_blackrock(file_fixed.path).header['start_time']
        t1 = localize_blackrock(file_moving.path).header['start_time']

        events0 = read_events_from_ephys(file_fixed, db=parent.db)
        events1 = read_events_from_ephys(file_moving, db=parent.db)

        self.offset_clock = (t1 - t0).total_seconds()

        layout = QGridLayout(self)
        layout.addWidget(make_table(events0), 1, 1)
        layout.addWidget(make_table(events1), 1, 2)

        self.evt_fixed = QDoubleSpinBox()
        self.evt_fixed.setDecimals(3)
        self.evt_fixed.setRange(-1e8, 1e8)
        self.evt_moving = QDoubleSpinBox()
        self.evt_moving.setDecimals(3)
        self.evt_moving.setRange(-1e8, 1e8)

        self.evt_fixed.editingFinished.connect(self.compute_offset)
        self.evt_moving.editingFinished.connect(self.compute_offset)
        layout.addWidget(self.evt_fixed, 2, 1)
        layout.addWidget(self.evt_moving, 2, 2)

        self.offset = QLabel('(offset will be computed)')
        layout.addWidget(self.offset, 3, 1)

        bbox = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
        bbox.accepted.connect(self.accept)
        bbox.rejected.connect(self.reject)
        layout.addWidget(bbox, 3, 2)

        self.setLayout(layout)

    def compute_offset(self):

        offset = (self.evt_moving.value() - self.evt_fixed.value()) + self.offset_clock
        self.offset.setText(f'{offset:0.3f}s')


class EditElectrodes(QDialog):

    def __init__(self, parent, data):
        super().__init__(parent)
        self.setWindowModality(Qt.WindowModal)

        self.parameter = QComboBox()
        for n in data.dtype.names:
            if n in ('name', 'x', 'y', 'z'):
                continue
            self.parameter.addItem(n)

        self.value = QLineEdit()
        self.value.setFixedWidth(800)

        bbox = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
        bbox.accepted.connect(self.accept)
        bbox.rejected.connect(self.reject)

        layout_file = QHBoxLayout()
        layout_file.addWidget(self.parameter)
        layout_file.addWidget(self.value)

        layout = QVBoxLayout()
        layout.addLayout(layout_file)
        layout.addWidget(bbox)

        self.setLayout(layout)


def _prepare_values(run, info):
    run_duration = run.duration
    if run_duration is None:
        run_duration = 0

    VALUES = [
        [
            '',
            'Current Values',
            'Imported Values'
        ],
        [
            'Start Time',
            strftime(run.start_time),
            strftime(info['start_time']),
        ],
        [
            'Duration',
            f'{run_duration:.3f} s',
            f'{info["duration"]:.3f} s',
        ],
        [
            '# Events',
            f'{len(run.events)}',
            f'{len(info["events"])}',
            ],
        ]

    return VALUES


def strftime(t):
    """This is the most accurate way to get milliseconds, without microseconds"""
    if t is None:
        return '(null)'
    else:
        return f'{t:%d/%m/%Y %H:%M:%S}.{t.microsecond / 1000:03.0f}'


class CompareEvents(QDialog):
    def __init__(self, parent, run, ephys_file):
        super().__init__(parent)

        self.info = read_info_from_ephys(parent.db, ephys_file)

        layout = QGridLayout(self)

        VALUES = _prepare_values(run, self.info)
        for i0, vals in enumerate(VALUES):
            for i1, value in enumerate(vals):
                label = QLabel(value)
                label.setAlignment(Qt.AlignRight | Qt.AlignVCenter)
                layout.addWidget(label, i0, i1)

        layout.addWidget(make_table(run.events), i0 + 1, 1)
        layout.addWidget(make_table(self.info['events']), i0 + 1, 2)

        self.old = QPushButton('Keep old events')
        self.old.clicked.connect(self.reject)
        self.new = QPushButton('Use new events')
        self.new.clicked.connect(self.accept)
        layout.addWidget(self.old, i0 + 2, 1)
        layout.addWidget(self.new, i0 + 2, 2)

        self.setLayout(layout)


def make_table(ev):
    t0 = QTableWidget()
    t0.horizontalHeader().setStretchLastSection(True)
    t0.setColumnCount(len(ev.dtype.names))
    t0.setHorizontalHeaderLabels(ev.dtype.names)
    t0.verticalHeader().setVisible(False)
    n_rows = len(ev)
    t0.setRowCount(n_rows)

    for i0, name in enumerate(ev.dtype.names):
        for i1 in range(n_rows):
            v = ev[name][i1]

            if issubdtype(ev.dtype[name].type, floating):
                v = f'{v:.3f}'

            elif issubdtype(ev.dtype[name].type, integer):
                v = f'{v}'

            item = QTableWidgetItem(v)
            item.setFlags(Qt.ItemIsSelectable | Qt.ItemIsEnabled)
            t0.setItem(i1, i0, item)

    table = t0
    table.setSizePolicy(QSizePolicy.Minimum, QSizePolicy.Expanding)

    table.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)

    table.resizeColumnsToContents()
    table.setFixedWidth(table.horizontalHeader().length() + table.verticalHeader().width())

    return t0


class Popup_Experimenters(QPushButton):

    def __init__(self, run, parent):
        self.run = run
        super().__init__(parent)
        self.set_title()

        self.menu = QMenu(self)
        current_experimenters = run.experimenters
        for name in list_experimenters(parent.db):
            action = QAction(name, self)
            action.setCheckable(True)
            if name in current_experimenters:
                action.setChecked(True)
            action.toggled.connect(self.action_toggle)
            self.menu.addAction(action)
        self.setMenu(self.menu)

    def action_toggle(self, checked):

        names = []
        for action in self.menu.actions():
            if action.isChecked():
                names.append(action.text())

        self.run.experimenters = names

        self.set_title()
        self.showMenu()

    def set_title(self):
        self.setText(', '.join(self.run.experimenters))


class Popup_Protocols(QPushButton):

    def __init__(self, run, parent):
        self.run = run
        super().__init__(parent)
        self.set_title()

        subject = run.session.subject

        current_protocols = [metc.id for metc in run.list_protocols()]
        self.menu = QMenu(self)
        for metc in subject.list_protocols():
            action = QAction(_protocol_name(metc), self)
            action.setCheckable(True)
            if metc.id in current_protocols:
                action.setChecked(True)
            action.toggled.connect(partial(self.action_toggle, metc=metc))
            self.menu.addAction(action)
        self.setMenu(self.menu)

    def action_toggle(self, checked, metc):
        if checked:
            self.run.attach_protocol(metc)
        else:
            self.run.detach_protocol(metc)

        self.set_title()
        self.showMenu()

    def set_title(self):
        self.setText(', '.join(_protocol_name(x) for x in self.run.list_protocols()))


class Popup_IntendedFor(QPushButton):

    def __init__(self, run, parent):
        self.run = run
        super().__init__(parent)
        self.set_title()

        current_targets = [x.id for x in run.intendedfor]

        self.menu = QMenu(self)
        for i, one_run in enumerate(run.session.list_runs()):
            if one_run.id == run.id:
                continue
            name = f'#{i + 1: 2d}: {one_run.task_name}'
            action = QAction(name, self)
            action.setCheckable(True)
            if one_run.id in current_targets:
                action.setChecked(True)
            action.toggled.connect(partial(self.action_toggle, target=one_run))
            self.menu.addAction(action)

        self.setMenu(self.menu)

    def action_toggle(self, checked, target):

        current_targets = self.run.intendedfor
        if checked:
            current_targets.append(target)
        else:
            current_targets = [x for x in current_targets if x.id != target.id]

        self.run.intendedfor = current_targets

        self.set_title()
        self.showMenu()

    def set_title(self):
        len_intendedfor = len(self.run.intendedfor)
        if len_intendedfor == 1:
            plural = ''
        else:
            plural = 's'

        self.setText(f'({len_intendedfor} target task{plural})')


class AccessDatabase(QDialog):

    def __init__(self, parent):
        super().__init__(parent)
        self.setWindowModality(Qt.WindowModal)

        bbox = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
        bbox.accepted.connect(self.accept)
        bbox.rejected.connect(self.reject)

        self.db_name = QLineEdit()
        self.hostname = QLineEdit()
        self.hostname.setText('127.0.0.1')
        self.username = QLineEdit()
        self.password = QLineEdit()
        self.password.setEchoMode(QLineEdit.Password)

        formlayout = QFormLayout()
        formlayout.addRow('Database Name', self.db_name)
        formlayout.addRow('Hostname', self.hostname)
        formlayout.addRow('Username', self.username)
        formlayout.addRow('Password', self.password)

        layout = QVBoxLayout()
        layout.addLayout(formlayout)
        layout.addWidget(bbox)

        self.setLayout(layout)


def parse_accessdatabase(parent):
    self = AccessDatabase(parent)
    result = self.exec()
    DB_ARGS = {}
    if result:
        db_name = self.db_name.text()
        if db_name == '':
            return None
        else:
            DB_ARGS['db_name'] = db_name
        DB_ARGS['username'] = self.username.text()
        DB_ARGS['password'] = self.password.text()
        DB_ARGS['hostname'] = self.hostname.text()

        return DB_ARGS

Functions

def make_table(ev)
Expand source code
def make_table(ev):
    t0 = QTableWidget()
    t0.horizontalHeader().setStretchLastSection(True)
    t0.setColumnCount(len(ev.dtype.names))
    t0.setHorizontalHeaderLabels(ev.dtype.names)
    t0.verticalHeader().setVisible(False)
    n_rows = len(ev)
    t0.setRowCount(n_rows)

    for i0, name in enumerate(ev.dtype.names):
        for i1 in range(n_rows):
            v = ev[name][i1]

            if issubdtype(ev.dtype[name].type, floating):
                v = f'{v:.3f}'

            elif issubdtype(ev.dtype[name].type, integer):
                v = f'{v}'

            item = QTableWidgetItem(v)
            item.setFlags(Qt.ItemIsSelectable | Qt.ItemIsEnabled)
            t0.setItem(i1, i0, item)

    table = t0
    table.setSizePolicy(QSizePolicy.Minimum, QSizePolicy.Expanding)

    table.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)

    table.resizeColumnsToContents()
    table.setFixedWidth(table.horizontalHeader().length() + table.verticalHeader().width())

    return t0
def parse_accessdatabase(parent)
Expand source code
def parse_accessdatabase(parent):
    self = AccessDatabase(parent)
    result = self.exec()
    DB_ARGS = {}
    if result:
        db_name = self.db_name.text()
        if db_name == '':
            return None
        else:
            DB_ARGS['db_name'] = db_name
        DB_ARGS['username'] = self.username.text()
        DB_ARGS['password'] = self.password.text()
        DB_ARGS['hostname'] = self.hostname.text()

        return DB_ARGS
def strftime(t)

This is the most accurate way to get milliseconds, without microseconds

Expand source code
def strftime(t):
    """This is the most accurate way to get milliseconds, without microseconds"""
    if t is None:
        return '(null)'
    else:
        return f'{t:%d/%m/%Y %H:%M:%S}.{t.microsecond / 1000:03.0f}'

Classes

class AccessDatabase (parent)

QDialog(parent: QWidget = None, flags: Union[Qt.WindowFlags, Qt.WindowType] = Qt.WindowFlags())

Expand source code
class AccessDatabase(QDialog):

    def __init__(self, parent):
        super().__init__(parent)
        self.setWindowModality(Qt.WindowModal)

        bbox = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
        bbox.accepted.connect(self.accept)
        bbox.rejected.connect(self.reject)

        self.db_name = QLineEdit()
        self.hostname = QLineEdit()
        self.hostname.setText('127.0.0.1')
        self.username = QLineEdit()
        self.password = QLineEdit()
        self.password.setEchoMode(QLineEdit.Password)

        formlayout = QFormLayout()
        formlayout.addRow('Database Name', self.db_name)
        formlayout.addRow('Hostname', self.hostname)
        formlayout.addRow('Username', self.username)
        formlayout.addRow('Password', self.password)

        layout = QVBoxLayout()
        layout.addLayout(formlayout)
        layout.addWidget(bbox)

        self.setLayout(layout)

Ancestors

  • PyQt5.QtWidgets.QDialog
  • PyQt5.QtWidgets.QWidget
  • PyQt5.QtCore.QObject
  • sip.wrapper
  • PyQt5.QtGui.QPaintDevice
  • sip.simplewrapper
class CalculateOffset (parent, file_fixed, file_moving)

QDialog(parent: QWidget = None, flags: Union[Qt.WindowFlags, Qt.WindowType] = Qt.WindowFlags())

Expand source code
class CalculateOffset(QDialog):
    def __init__(self, parent, file_fixed, file_moving):
        super().__init__(parent)

        t0 = localize_blackrock(file_fixed.path).header['start_time']
        t1 = localize_blackrock(file_moving.path).header['start_time']

        events0 = read_events_from_ephys(file_fixed, db=parent.db)
        events1 = read_events_from_ephys(file_moving, db=parent.db)

        self.offset_clock = (t1 - t0).total_seconds()

        layout = QGridLayout(self)
        layout.addWidget(make_table(events0), 1, 1)
        layout.addWidget(make_table(events1), 1, 2)

        self.evt_fixed = QDoubleSpinBox()
        self.evt_fixed.setDecimals(3)
        self.evt_fixed.setRange(-1e8, 1e8)
        self.evt_moving = QDoubleSpinBox()
        self.evt_moving.setDecimals(3)
        self.evt_moving.setRange(-1e8, 1e8)

        self.evt_fixed.editingFinished.connect(self.compute_offset)
        self.evt_moving.editingFinished.connect(self.compute_offset)
        layout.addWidget(self.evt_fixed, 2, 1)
        layout.addWidget(self.evt_moving, 2, 2)

        self.offset = QLabel('(offset will be computed)')
        layout.addWidget(self.offset, 3, 1)

        bbox = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
        bbox.accepted.connect(self.accept)
        bbox.rejected.connect(self.reject)
        layout.addWidget(bbox, 3, 2)

        self.setLayout(layout)

    def compute_offset(self):

        offset = (self.evt_moving.value() - self.evt_fixed.value()) + self.offset_clock
        self.offset.setText(f'{offset:0.3f}s')

Ancestors

  • PyQt5.QtWidgets.QDialog
  • PyQt5.QtWidgets.QWidget
  • PyQt5.QtCore.QObject
  • sip.wrapper
  • PyQt5.QtGui.QPaintDevice
  • sip.simplewrapper

Methods

def compute_offset(self)
Expand source code
def compute_offset(self):

    offset = (self.evt_moving.value() - self.evt_fixed.value()) + self.offset_clock
    self.offset.setText(f'{offset:0.3f}s')
class CompareEvents (parent, run, ephys_file)

QDialog(parent: QWidget = None, flags: Union[Qt.WindowFlags, Qt.WindowType] = Qt.WindowFlags())

Expand source code
class CompareEvents(QDialog):
    def __init__(self, parent, run, ephys_file):
        super().__init__(parent)

        self.info = read_info_from_ephys(parent.db, ephys_file)

        layout = QGridLayout(self)

        VALUES = _prepare_values(run, self.info)
        for i0, vals in enumerate(VALUES):
            for i1, value in enumerate(vals):
                label = QLabel(value)
                label.setAlignment(Qt.AlignRight | Qt.AlignVCenter)
                layout.addWidget(label, i0, i1)

        layout.addWidget(make_table(run.events), i0 + 1, 1)
        layout.addWidget(make_table(self.info['events']), i0 + 1, 2)

        self.old = QPushButton('Keep old events')
        self.old.clicked.connect(self.reject)
        self.new = QPushButton('Use new events')
        self.new.clicked.connect(self.accept)
        layout.addWidget(self.old, i0 + 2, 1)
        layout.addWidget(self.new, i0 + 2, 2)

        self.setLayout(layout)

Ancestors

  • PyQt5.QtWidgets.QDialog
  • PyQt5.QtWidgets.QWidget
  • PyQt5.QtCore.QObject
  • sip.wrapper
  • PyQt5.QtGui.QPaintDevice
  • sip.simplewrapper
class EditElectrodes (parent, data)

QDialog(parent: QWidget = None, flags: Union[Qt.WindowFlags, Qt.WindowType] = Qt.WindowFlags())

Expand source code
class EditElectrodes(QDialog):

    def __init__(self, parent, data):
        super().__init__(parent)
        self.setWindowModality(Qt.WindowModal)

        self.parameter = QComboBox()
        for n in data.dtype.names:
            if n in ('name', 'x', 'y', 'z'):
                continue
            self.parameter.addItem(n)

        self.value = QLineEdit()
        self.value.setFixedWidth(800)

        bbox = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
        bbox.accepted.connect(self.accept)
        bbox.rejected.connect(self.reject)

        layout_file = QHBoxLayout()
        layout_file.addWidget(self.parameter)
        layout_file.addWidget(self.value)

        layout = QVBoxLayout()
        layout.addLayout(layout_file)
        layout.addWidget(bbox)

        self.setLayout(layout)

Ancestors

  • PyQt5.QtWidgets.QDialog
  • PyQt5.QtWidgets.QWidget
  • PyQt5.QtCore.QObject
  • sip.wrapper
  • PyQt5.QtGui.QPaintDevice
  • sip.simplewrapper
class NewFile (parent, file_obj=None, level_obj=None, filename=None)

QDialog(parent: QWidget = None, flags: Union[Qt.WindowFlags, Qt.WindowType] = Qt.WindowFlags())

Expand source code
class NewFile(QDialog):

    def __init__(self, parent, file_obj=None, level_obj=None, filename=None):
        super().__init__(parent)
        self.setWindowModality(Qt.WindowModal)

        self.level = QComboBox()
        self.level.addItems([level[:-1].capitalize() for level in LEVELS])

        if level_obj is not None:
            self.level.setEnabled(False)  # do not allow changing level here
            self.level.setCurrentText(level_obj.t.capitalize())

        self.filepath = QLineEdit()
        self.filepath.setFixedWidth(800)
        self.filepath.editingFinished.connect(self.set_filetype)
        browse = QPushButton('Browse ...')
        browse.clicked.connect(self.browse)
        self.format = QComboBox()
        self.format.addItems(['Unknown', ] + lookup_allowed_values(parent.db['db'], 'files', 'format'))

        bbox = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
        bbox.accepted.connect(self.accept)
        bbox.rejected.connect(self.reject)

        layout_file = QHBoxLayout()
        layout_file.addWidget(self.filepath)
        layout_file.addWidget(browse)

        layout = QVBoxLayout()
        layout.addWidget(self.level)
        layout.addLayout(layout_file)
        layout.addWidget(self.format)
        layout.addWidget(bbox)

        self.setLayout(layout)

        if file_obj is not None:
            # self.level.setCurrentText(file_obj)
            self.filepath.setText(str(file_obj.path))
            self.format.setCurrentText(file_obj.format)

        if filename is not None:
            self.filepath.setText(filename)
            self.set_filetype()
            self.filepath.setEnabled(False)
            self.set_filetype(filename)

    def browse(self):
        filename, _ = QFileDialog.getOpenFileName(self, 'Select File')

        if filename:
            self.filepath.setText(filename)
            self.set_filetype(filename)

    def set_filetype(self, filename=None):
        if filename is None:
            filename = Path(self.filepath.text()).resolve()

        try:
            filetype = parse_filetype(filename)

        except ValueError as err:
            lg.debug(err)
            print(err)

        else:
            self.format.setCurrentText(filetype)

Ancestors

  • PyQt5.QtWidgets.QDialog
  • PyQt5.QtWidgets.QWidget
  • PyQt5.QtCore.QObject
  • sip.wrapper
  • PyQt5.QtGui.QPaintDevice
  • sip.simplewrapper

Methods

def browse(self)
Expand source code
def browse(self):
    filename, _ = QFileDialog.getOpenFileName(self, 'Select File')

    if filename:
        self.filepath.setText(filename)
        self.set_filetype(filename)
def set_filetype(self, filename=None)
Expand source code
def set_filetype(self, filename=None):
    if filename is None:
        filename = Path(self.filepath.text()).resolve()

    try:
        filetype = parse_filetype(filename)

    except ValueError as err:
        lg.debug(err)
        print(err)

    else:
        self.format.setCurrentText(filetype)
class Popup_Experimenters (run, parent)

QPushButton(parent: QWidget = None) QPushButton(str, parent: QWidget = None) QPushButton(QIcon, str, parent: QWidget = None)

Expand source code
class Popup_Experimenters(QPushButton):

    def __init__(self, run, parent):
        self.run = run
        super().__init__(parent)
        self.set_title()

        self.menu = QMenu(self)
        current_experimenters = run.experimenters
        for name in list_experimenters(parent.db):
            action = QAction(name, self)
            action.setCheckable(True)
            if name in current_experimenters:
                action.setChecked(True)
            action.toggled.connect(self.action_toggle)
            self.menu.addAction(action)
        self.setMenu(self.menu)

    def action_toggle(self, checked):

        names = []
        for action in self.menu.actions():
            if action.isChecked():
                names.append(action.text())

        self.run.experimenters = names

        self.set_title()
        self.showMenu()

    def set_title(self):
        self.setText(', '.join(self.run.experimenters))

Ancestors

  • PyQt5.QtWidgets.QPushButton
  • PyQt5.QtWidgets.QAbstractButton
  • PyQt5.QtWidgets.QWidget
  • PyQt5.QtCore.QObject
  • sip.wrapper
  • PyQt5.QtGui.QPaintDevice
  • sip.simplewrapper

Methods

def action_toggle(self, checked)
Expand source code
def action_toggle(self, checked):

    names = []
    for action in self.menu.actions():
        if action.isChecked():
            names.append(action.text())

    self.run.experimenters = names

    self.set_title()
    self.showMenu()
def set_title(self)
Expand source code
def set_title(self):
    self.setText(', '.join(self.run.experimenters))
class Popup_IntendedFor (run, parent)

QPushButton(parent: QWidget = None) QPushButton(str, parent: QWidget = None) QPushButton(QIcon, str, parent: QWidget = None)

Expand source code
class Popup_IntendedFor(QPushButton):

    def __init__(self, run, parent):
        self.run = run
        super().__init__(parent)
        self.set_title()

        current_targets = [x.id for x in run.intendedfor]

        self.menu = QMenu(self)
        for i, one_run in enumerate(run.session.list_runs()):
            if one_run.id == run.id:
                continue
            name = f'#{i + 1: 2d}: {one_run.task_name}'
            action = QAction(name, self)
            action.setCheckable(True)
            if one_run.id in current_targets:
                action.setChecked(True)
            action.toggled.connect(partial(self.action_toggle, target=one_run))
            self.menu.addAction(action)

        self.setMenu(self.menu)

    def action_toggle(self, checked, target):

        current_targets = self.run.intendedfor
        if checked:
            current_targets.append(target)
        else:
            current_targets = [x for x in current_targets if x.id != target.id]

        self.run.intendedfor = current_targets

        self.set_title()
        self.showMenu()

    def set_title(self):
        len_intendedfor = len(self.run.intendedfor)
        if len_intendedfor == 1:
            plural = ''
        else:
            plural = 's'

        self.setText(f'({len_intendedfor} target task{plural})')

Ancestors

  • PyQt5.QtWidgets.QPushButton
  • PyQt5.QtWidgets.QAbstractButton
  • PyQt5.QtWidgets.QWidget
  • PyQt5.QtCore.QObject
  • sip.wrapper
  • PyQt5.QtGui.QPaintDevice
  • sip.simplewrapper

Methods

def action_toggle(self, checked, target)
Expand source code
def action_toggle(self, checked, target):

    current_targets = self.run.intendedfor
    if checked:
        current_targets.append(target)
    else:
        current_targets = [x for x in current_targets if x.id != target.id]

    self.run.intendedfor = current_targets

    self.set_title()
    self.showMenu()
def set_title(self)
Expand source code
def set_title(self):
    len_intendedfor = len(self.run.intendedfor)
    if len_intendedfor == 1:
        plural = ''
    else:
        plural = 's'

    self.setText(f'({len_intendedfor} target task{plural})')
class Popup_Protocols (run, parent)

QPushButton(parent: QWidget = None) QPushButton(str, parent: QWidget = None) QPushButton(QIcon, str, parent: QWidget = None)

Expand source code
class Popup_Protocols(QPushButton):

    def __init__(self, run, parent):
        self.run = run
        super().__init__(parent)
        self.set_title()

        subject = run.session.subject

        current_protocols = [metc.id for metc in run.list_protocols()]
        self.menu = QMenu(self)
        for metc in subject.list_protocols():
            action = QAction(_protocol_name(metc), self)
            action.setCheckable(True)
            if metc.id in current_protocols:
                action.setChecked(True)
            action.toggled.connect(partial(self.action_toggle, metc=metc))
            self.menu.addAction(action)
        self.setMenu(self.menu)

    def action_toggle(self, checked, metc):
        if checked:
            self.run.attach_protocol(metc)
        else:
            self.run.detach_protocol(metc)

        self.set_title()
        self.showMenu()

    def set_title(self):
        self.setText(', '.join(_protocol_name(x) for x in self.run.list_protocols()))

Ancestors

  • PyQt5.QtWidgets.QPushButton
  • PyQt5.QtWidgets.QAbstractButton
  • PyQt5.QtWidgets.QWidget
  • PyQt5.QtCore.QObject
  • sip.wrapper
  • PyQt5.QtGui.QPaintDevice
  • sip.simplewrapper

Methods

def action_toggle(self, checked, metc)
Expand source code
def action_toggle(self, checked, metc):
    if checked:
        self.run.attach_protocol(metc)
    else:
        self.run.detach_protocol(metc)

    self.set_title()
    self.showMenu()
def set_title(self)
Expand source code
def set_title(self):
    self.setText(', '.join(_protocol_name(x) for x in self.run.list_protocols()))