Source code for guidata.dataset.qtitemwidgets

# -*- coding: utf-8 -*-
#
# Copyright © 2009-2010 CEA
# Pierre Raybaut
# Licensed under the terms of the CECILL License
# (see guidata/__init__.py for details)

"""
dataset.qtitemwidgets
=====================

Widget factories used to edit data items
(factory registration is done in guidata.dataset.qtwidgets)
(data item types are defined in guidata.dataset.datatypes)

There is one widget type for each data item type.
Example: ChoiceWidget <--> ChoiceItem, ImageChoiceItem
"""

import os
import os.path as osp
import sys
import numpy
import collections.abc
import datetime

from qtpy.QtWidgets import (
    QHBoxLayout,
    QGridLayout,
    QColorDialog,
    QPushButton,
    QLineEdit,
    QCheckBox,
    QComboBox,
    QTabWidget,
    QGroupBox,
    QDateTimeEdit,
    QLabel,
    QTextEdit,
    QFrame,
    QDateEdit,
    QSlider,
    QRadioButton,
    QVBoxLayout,
)
from qtpy.QtGui import QColor, QIcon, QPixmap
from qtpy.QtCore import Qt, Signal
from qtpy.compat import getexistingdirectory

from guidata.utils import update_dataset, restore_dataset, utf8_to_unicode
from guidata.qthelpers import text_to_qcolor, get_std_icon
from guidata.configtools import get_icon, get_image_layout, get_image_file_path
from guidata.config import _
from guidata.widgets.arrayeditor import ArrayEditor

# ========================== <!> IMPORTANT <!> =================================
#
# In this module, `item` is an instance of DataItemVariable (not DataItem)
# (see guidata.datatypes for details)
#
# ========================== <!> IMPORTANT <!> =================================

# XXX: consider providing an interface here...


[docs]class AbstractDataSetWidget(object): """ Base class for 'widgets' handled by `DataSetEditLayout` and it's derived classes. This is a generic representation of an input (or display) widget that has a label and one or more entry field. `DataSetEditLayout` uses a registry of *Item* to *Widget* mapping in order to automatically create a GUI for a `DataSet` structure """ READ_ONLY = False def __init__(self, item, parent_layout): """Derived constructors should create the necessary widgets The base class keeps a reference to item and parent """ self.item = item self.parent_layout = parent_layout self.group = None # Layout/Widget grouping items self.label = None self.build_mode = False
[docs] def place_label(self, layout, row, column): """ Place item label on layout at specified position (row, column) """ label_text = self.item.get_prop_value("display", "label") unit = self.item.get_prop_value("display", "unit", "") if unit and not self.READ_ONLY: label_text += " (%s)" % unit self.label = QLabel(label_text) self.label.setToolTip(self.item.get_help()) layout.addWidget(self.label, row, column)
[docs] def place_on_grid( self, layout, row, label_column, widget_column, row_span=1, column_span=1 ): """ Place widget on layout at specified position """ self.place_label(layout, row, label_column) layout.addWidget(self.group, row, widget_column, row_span, column_span)
[docs] def is_active(self): """ Return True if associated item is active """ return self.item.get_prop_value("display", "active", True)
[docs] def check(self): """ Item validator """ return True
[docs] def set(self): """ Update data item value from widget contents """ # XXX: consider using item.set instead of item.set_from_string... self.item.set_from_string(self.value())
[docs] def get(self): """ Update widget contents from data item value """ pass
[docs] def value(self): """ Returns the widget's current value """ return None
[docs] def set_state(self): """ Update the visual status of the widget """ active = self.is_active() if self.group: self.group.setEnabled(active) if self.label: self.label.setEnabled(active)
[docs] def notify_value_change(self): """ Notify parent layout that widget value has changed """ if not self.build_mode: self.parent_layout.widget_value_changed()
[docs]class GroupWidget(AbstractDataSetWidget): """ GroupItem widget """ def __init__(self, item, parent_layout): super(GroupWidget, self).__init__(item, parent_layout) embedded = item.get_prop_value("display", "embedded", False) if not embedded: self.group = QGroupBox(item.get_prop_value("display", "label")) else: self.group = QFrame() self.layout = QGridLayout() EditLayoutClass = parent_layout.__class__ self.edit = EditLayoutClass( self.group, item.instance, self.layout, item.item.group, change_callback=self.notify_value_change, ) self.group.setLayout(self.layout)
[docs] def get(self): """Override AbstractDataSetWidget method""" self.edit.update_widgets()
[docs] def set(self): """Override AbstractDataSetWidget method""" self.edit.accept_changes()
[docs] def check(self): """Override AbstractDataSetWidget method""" return self.edit.check_all_values()
[docs] def place_on_grid( self, layout, row, label_column, widget_column, row_span=1, column_span=1 ): """Override AbstractDataSetWidget method""" layout.addWidget(self.group, row, label_column, row_span, column_span + 1)
[docs]class TabGroupWidget(AbstractDataSetWidget): def __init__(self, item, parent_layout): super(TabGroupWidget, self).__init__(item, parent_layout) self.tabs = QTabWidget() items = item.item.group self.widgets = [] for item in items: if item.get_prop_value("display", parent_layout.instance, "hide", False): continue item.set_prop("display", embedded=True) widget = parent_layout.build_widget(item) frame = QFrame() label = widget.item.get_prop_value("display", "label") icon = widget.item.get_prop_value("display", "icon", None) if icon is not None: self.tabs.addTab(frame, get_icon(icon), label) else: self.tabs.addTab(frame, label) layout = QGridLayout() layout.setAlignment(Qt.AlignTop) frame.setLayout(layout) widget.place_on_grid(layout, 0, 0, 1) try: widget.get() except Exception: print("Error building item :", item.item._name) raise self.widgets.append(widget)
[docs] def get(self): """Override AbstractDataSetWidget method""" for widget in self.widgets: widget.get()
[docs] def set(self): """Override AbstractDataSetWidget method""" for widget in self.widgets: widget.set()
[docs] def check(self): """Override AbstractDataSetWidget method""" return True
[docs] def place_on_grid( self, layout, row, label_column, widget_column, row_span=1, column_span=1 ): """Override AbstractDataSetWidget method""" layout.addWidget(self.tabs, row, label_column, row_span, column_span + 1)
[docs]class LineEditWidget(AbstractDataSetWidget): """ QLineEdit-based widget """ def __init__(self, item, parent_layout): super(LineEditWidget, self).__init__(item, parent_layout) self.edit = self.group = QLineEdit() self.edit.setToolTip(item.get_help()) if hasattr(item, "min_equals_max") and item.min_equals_max(): if item.check_item(): self.edit.setEnabled(False) self.edit.setToolTip(_("Value is forced to %d") % item.get_max()) self.edit.textChanged.connect(self.line_edit_changed)
[docs] def get(self): """Override AbstractDataSetWidget method""" value = self.item.get() old_value = str(self.value()) if value is not None: if isinstance(value, QColor): # if item is a ColorItem object value = value.name() uvalue = utf8_to_unicode(value) if uvalue != old_value: self.edit.setText(utf8_to_unicode(value)) else: self.line_edit_changed(value)
[docs] def line_edit_changed(self, qvalue): """QLineEdit validator""" if qvalue is not None: value = self.item.from_string(str(qvalue)) else: value = None if not self.item.check_value(value): self.edit.setStyleSheet("background-color:rgb(255, 175, 90);") else: self.edit.setStyleSheet("") cb = self.item.get_prop_value("display", "callback", None) if cb is not None: if self.build_mode: self.set() else: self.parent_layout.update_dataitems() cb(self.item.instance, self.item.item, value) self.parent_layout.update_widgets(except_this_one=self) self.update(value) self.notify_value_change()
[docs] def update(self, value): """Override AbstractDataSetWidget method""" cb = self.item.get_prop_value("display", "value_callback", None) if cb is not None: cb(value)
[docs] def value(self): return str(self.edit.text())
[docs] def check(self): """Override AbstractDataSetWidget method""" value = self.item.from_string(str(self.edit.text())) return self.item.check_value(value)
[docs]class TextEditWidget(AbstractDataSetWidget): """ QTextEdit-based widget """ def __init__(self, item, parent_layout): super(TextEditWidget, self).__init__(item, parent_layout) self.edit = self.group = QTextEdit() self.edit.setToolTip(item.get_help()) if hasattr(item, "min_equals_max") and item.min_equals_max(): if item.check_item(): self.edit.setEnabled(False) self.edit.setToolTip(_("Value is forced to %d") % item.get_max()) self.edit.textChanged.connect(self.text_changed) def __get_text(self): """Get QTextEdit text, replacing UTF-8 EOL chars by os.linesep""" return str(self.edit.toPlainText()).replace("\u2029", os.linesep)
[docs] def get(self): """Override AbstractDataSetWidget method""" value = self.item.get() if value is not None: self.edit.setPlainText(utf8_to_unicode(value)) self.text_changed()
[docs] def text_changed(self): """QLineEdit validator""" value = self.item.from_string(self.__get_text()) if not self.item.check_value(value): self.edit.setStyleSheet("background-color:rgb(255, 175, 90);") else: self.edit.setStyleSheet("") self.update(value) self.notify_value_change()
[docs] def update(self, value): """Override AbstractDataSetWidget method""" pass
[docs] def value(self): return self.edit.toPlainText()
[docs] def check(self): """Override AbstractDataSetWidget method""" value = self.item.from_string(self.__get_text()) return self.item.check_value(value)
[docs]class CheckBoxWidget(AbstractDataSetWidget): """ BoolItem widget """ def __init__(self, item, parent_layout): super(CheckBoxWidget, self).__init__(item, parent_layout) self.checkbox = QCheckBox(self.item.get_prop_value("display", "text")) self.checkbox.setToolTip(item.get_help()) self.group = self.checkbox self.store = self.item.get_prop("display", "store", None) self.checkbox.stateChanged.connect(self.state_changed)
[docs] def get(self): """Override AbstractDataSetWidget method""" value = self.item.get() if value is not None: self.checkbox.setChecked(value)
[docs] def set(self): """Override AbstractDataSetWidget method""" self.item.set(self.value())
[docs] def value(self): return self.checkbox.isChecked()
[docs] def place_on_grid( self, layout, row, label_column, widget_column, row_span=1, column_span=1 ): """Override AbstractDataSetWidget method""" if not self.item.get_prop_value("display", "label"): widget_column = label_column column_span += 1 else: self.place_label(layout, row, label_column) layout.addWidget(self.group, row, widget_column, row_span, column_span)
def state_changed(self, state): self.notify_value_change() if self.store: self.do_store(state) def do_store(self, state): self.store.set(self.item.instance, self.item.item, state) self.parent_layout.refresh_widgets()
[docs]class DateWidget(AbstractDataSetWidget): """ DateItem widget """ def __init__(self, item, parent_layout): super(DateWidget, self).__init__(item, parent_layout) self.dateedit = self.group = QDateEdit() self.dateedit.setToolTip(item.get_help()) self.dateedit.dateTimeChanged.connect(lambda value: self.notify_value_change())
[docs] def get(self): """Override AbstractDataSetWidget method""" value = self.item.get() if value: if not isinstance(value, datetime.date): value = datetime.date.fromordinal(value) self.dateedit.setDate(value)
[docs] def set(self): """Override AbstractDataSetWidget method""" self.item.set(self.value())
[docs] def value(self): try: return self.dateedit.date().toPyDate() except AttributeError: return self.dateedit.dateTime().toPython().date() # PySide
[docs]class DateTimeWidget(AbstractDataSetWidget): """ DateTimeItem widget """ def __init__(self, item, parent_layout): super(DateTimeWidget, self).__init__(item, parent_layout) self.dateedit = self.group = QDateTimeEdit() self.dateedit.setCalendarPopup(True) self.dateedit.setToolTip(item.get_help()) self.dateedit.dateTimeChanged.connect(lambda value: self.notify_value_change())
[docs] def get(self): """Override AbstractDataSetWidget method""" value = self.item.get() if value: if not isinstance(value, datetime.datetime): value = datetime.datetime.fromtimestamp(value) self.dateedit.setDateTime(value)
[docs] def set(self): """Override AbstractDataSetWidget method""" self.item.set(self.value())
[docs] def value(self): try: return self.dateedit.dateTime().toPyDateTime() except AttributeError: return self.dateedit.dateTime().toPython() # PySide
[docs]class GroupLayout(QHBoxLayout): def __init__(self): QHBoxLayout.__init__(self) self.widgets = []
[docs] def addWidget(self, widget): QHBoxLayout.addWidget(self, widget) self.widgets.append(widget)
[docs] def setEnabled(self, state): for widget in self.widgets: widget.setEnabled(state)
class HLayoutMixin(object): def __init__(self, item, parent_layout): super(HLayoutMixin, self).__init__(item, parent_layout) old_group = self.group self.group = GroupLayout() self.group.addWidget(old_group) def place_on_grid( self, layout, row, label_column, widget_column, row_span=1, column_span=1 ): """Override AbstractDataSetWidget method""" self.place_label(layout, row, label_column) layout.addLayout(self.group, row, widget_column, row_span, column_span)
[docs]class ColorWidget(HLayoutMixin, LineEditWidget): """ ColorItem widget """ def __init__(self, item, parent_layout): super(ColorWidget, self).__init__(item, parent_layout) self.button = QPushButton("") self.button.setMaximumWidth(32) self.button.clicked.connect(self.select_color) self.group.addWidget(self.button)
[docs] def update(self, value): """Reimplement LineEditWidget method""" LineEditWidget.update(self, value) color = text_to_qcolor(value) if color.isValid(): bitmap = QPixmap(16, 16) bitmap.fill(color) icon = QIcon(bitmap) else: icon = get_icon("not_found") self.button.setIcon(icon)
[docs] def select_color(self): """Open a color selection dialog box""" color = text_to_qcolor(self.edit.text()) if not color.isValid(): color = Qt.gray color = QColorDialog.getColor(color, self.parent_layout.parent) if color.isValid(): value = color.name() self.edit.setText(value) self.update(value) self.notify_value_change()
[docs]class SliderWidget(HLayoutMixin, LineEditWidget): """ IntItem with Slider """ DATA_TYPE = int def __init__(self, item, parent_layout): super(SliderWidget, self).__init__(item, parent_layout) self.slider = self.vmin = self.vmax = None if item.get_prop_value("display", "slider"): self.vmin = item.get_prop_value("data", "min") self.vmax = item.get_prop_value("data", "max") assert ( self.vmin is not None and self.vmax is not None ), "SliderWidget requires that item min/max have been defined" self.slider = QSlider() self.slider.setOrientation(Qt.Horizontal) self.setup_slider(item) self.slider.valueChanged.connect(self.value_changed) self.group.addWidget(self.slider) def value_to_slider(self, value): return value def slider_to_value(self, value): return value def setup_slider(self, item): self.slider.setRange(self.vmin, self.vmax)
[docs] def update(self, value): """Reimplement LineEditWidget method""" LineEditWidget.update(self, value) if self.slider is not None and isinstance(value, self.DATA_TYPE): self.slider.blockSignals(True) self.slider.setValue(self.value_to_slider(value)) self.slider.blockSignals(False)
[docs] def value_changed(self, ivalue): """Update the lineedit""" value = str(self.slider_to_value(ivalue)) self.edit.setText(value) self.update(value)
[docs]class FloatSliderWidget(SliderWidget): """ FloatItem with Slider """ DATA_TYPE = float def value_to_slider(self, value): return int((value - self.vmin) * 100 / (self.vmax - self.vmin)) def slider_to_value(self, value): return value * (self.vmax - self.vmin) / 100 + self.vmin def setup_slider(self, item): self.slider.setRange(0, 100)
def _get_child_title_func(ancestor): previous_ancestor = None while True: try: if previous_ancestor is ancestor: break return ancestor.child_title except AttributeError: previous_ancestor = ancestor ancestor = ancestor.parent() return lambda item: ""
[docs]class FileWidget(HLayoutMixin, LineEditWidget): """ File path item widget """ def __init__(self, item, parent_layout, filedialog): super(FileWidget, self).__init__(item, parent_layout) self.filedialog = filedialog button = QPushButton() fmt = item.get_prop_value("data", "formats") button.setIcon(get_icon("%s.png" % fmt[0].lower(), default="file.png")) button.clicked.connect(self.select_file) self.group.addWidget(button) self.basedir = item.get_prop_value("data", "basedir") self.all_files_first = item.get_prop_value("data", "all_files_first")
[docs] def select_file(self): """Open a file selection dialog box""" fname = self.item.from_string(str(self.edit.text())) if isinstance(fname, list): fname = osp.dirname(fname[0]) parent = self.parent_layout.parent _temp = sys.stdout sys.stdout = None if len(fname) == 0: fname = self.basedir _formats = self.item.get_prop_value("data", "formats") formats = [str(format).lower() for format in _formats] filter_lines = [ (_("%s files") + " (*.%s)") % (format.upper(), format) for format in formats ] all_filter = _("All supported files") + " (*.%s)" % " *.".join(formats) if len(formats) > 1: if self.all_files_first: filter_lines.insert(0, all_filter) else: filter_lines.append(all_filter) if fname is None: fname = "" child_title = _get_child_title_func(parent) fname, _filter = self.filedialog( parent, child_title(self.item), fname, "\n".join(filter_lines) ) sys.stdout = _temp if fname: if isinstance(fname, list): fname = str(fname) self.edit.setText(fname)
[docs]class DirectoryWidget(HLayoutMixin, LineEditWidget): """ Directory path item widget """ def __init__(self, item, parent_layout): super(DirectoryWidget, self).__init__(item, parent_layout) button = QPushButton() button.setIcon(get_std_icon("DirOpenIcon")) button.clicked.connect(self.select_directory) self.group.addWidget(button)
[docs] def select_directory(self): """Open a directory selection dialog box""" value = self.item.from_string(str(self.edit.text())) parent = self.parent_layout.parent child_title = _get_child_title_func(parent) dname = getexistingdirectory(parent, child_title(self.item), value) if dname: self.edit.setText(dname)
[docs]class ChoiceWidget(AbstractDataSetWidget): """ Choice item widget """ def __init__(self, item, parent_layout): super(ChoiceWidget, self).__init__(item, parent_layout) self._first_call = True self.is_radio = item.get_prop_value("display", "radio") self.store = self.item.get_prop("display", "store", None) if self.is_radio: self.group = QGroupBox() self.group.setToolTip(item.get_help()) self.vbox = QVBoxLayout() self.group.setLayout(self.vbox) self._buttons = [] else: self.combobox = self.group = QComboBox() self.combobox.setToolTip(item.get_help()) self.combobox.currentIndexChanged.connect(self.index_changed) def index_changed(self, index): if self.store: self.store.set(self.item.instance, self.item.item, self.value()) self.parent_layout.refresh_widgets() cb = self.item.get_prop_value("display", "callback", None) if cb is not None: if self.build_mode: self.set() else: self.parent_layout.update_dataitems() cb(self.item.instance, self.item.item, self.value()) self.parent_layout.update_widgets(except_this_one=self) self.notify_value_change() def initialize_widget(self): if self.is_radio: for button in self._buttons: button.toggled.disconnect(self.index_changed) self.vbox.removeWidget(button) button.deleteLater() self._buttons = [] else: self.combobox.blockSignals(True) while self.combobox.count(): self.combobox.removeItem(0) _choices = self.item.get_prop_value("data", "choices") for key, lbl, img in _choices: if self.is_radio: button = QRadioButton(lbl, self.group) if img: if isinstance(img, str): if not osp.isfile(img): img = get_image_file_path(img) img = QIcon(img) elif isinstance(img, collections.abc.Callable): img = img(key) if self.is_radio: button.setIcon(img) else: self.combobox.addItem(img, lbl) elif not self.is_radio: self.combobox.addItem(lbl) if self.is_radio: self._buttons.append(button) self.vbox.addWidget(button) button.toggled.connect(self.index_changed) if not self.is_radio: self.combobox.blockSignals(False) def set_widget_value(self, idx): if self.is_radio: for button in self._buttons: button.blockSignals(True) self._buttons[idx].setChecked(True) for button in self._buttons: button.blockSignals(False) else: self.combobox.blockSignals(True) self.combobox.setCurrentIndex(idx) self.combobox.blockSignals(False) def get_widget_value(self): if self.is_radio: for index, widget in enumerate(self._buttons): if widget.isChecked(): return index else: return self.combobox.currentIndex()
[docs] def get(self): """Override AbstractDataSetWidget method""" self.initialize_widget() value = self.item.get() if value is not None: idx = 0 _choices = self.item.get_prop_value("data", "choices") for key, _val, _img in _choices: if key == value: break idx += 1 self.set_widget_value(idx) if self._first_call: self.index_changed(idx) self._first_call = False
[docs] def set(self): """Override AbstractDataSetWidget method""" try: value = self.value() except IndexError: return self.item.set(value)
[docs] def value(self): index = self.get_widget_value() choices = self.item.get_prop_value("data", "choices") return choices[index][0]
[docs]class MultipleChoiceWidget(AbstractDataSetWidget): """ Multiple choice item widget """ def __init__(self, item, parent_layout): super(MultipleChoiceWidget, self).__init__(item, parent_layout) self.groupbox = self.group = QGroupBox(item.get_prop_value("display", "label")) layout = QGridLayout() self.boxes = [] nx, ny = item.get_prop_value("display", "shape") cx, cy = 0, 0 _choices = item.get_prop_value("data", "choices") for _, choice, _img in _choices: checkbox = QCheckBox(choice) layout.addWidget(checkbox, cx, cy) if nx < 0: cy += 1 if cy >= ny: cy = 0 cx += 1 else: cx += 1 if cx >= nx: cx = 0 cy += 1 self.boxes.append(checkbox) self.groupbox.setLayout(layout)
[docs] def get(self): """Override AbstractDataSetWidget method""" value = self.item.get() _choices = self.item.get_prop_value("data", "choices") for (i, _choice, _img), checkbox in zip(_choices, self.boxes): if value is not None and i in value: checkbox.setChecked(True)
[docs] def set(self): """Override AbstractDataSetWidget method""" _choices = self.item.get_prop_value("data", "choices") choices = [_choices[i][0] for i in self.value()] self.item.set(choices)
[docs] def value(self): return [i for i, w in enumerate(self.boxes) if w.isChecked()]
[docs] def place_on_grid( self, layout, row, label_column, widget_column, row_span=1, column_span=1 ): """Override AbstractDataSetWidget method""" layout.addWidget(self.group, row, label_column, row_span, column_span + 1)
[docs]class FloatArrayWidget(AbstractDataSetWidget): """ FloatArrayItem widget """ def __init__(self, item, parent_layout): super(FloatArrayWidget, self).__init__(item, parent_layout) _label = item.get_prop_value("display", "label") self.groupbox = self.group = QGroupBox(_label) self.layout = QGridLayout() self.layout.setAlignment(Qt.AlignLeft) self.groupbox.setLayout(self.layout) self.first_line, self.dim_label = get_image_layout( "shape.png", _("Number of rows x Number of columns") ) edit_button = QPushButton(get_icon("arredit.png"), "") edit_button.setToolTip(_("Edit array contents")) edit_button.setMaximumWidth(32) self.first_line.addWidget(edit_button) self.layout.addLayout(self.first_line, 0, 0) self.min_line, self.min_label = get_image_layout( "min.png", _("Smallest element in array") ) self.layout.addLayout(self.min_line, 1, 0) self.max_line, self.max_label = get_image_layout( "max.png", _("Largest element in array") ) self.layout.addLayout(self.max_line, 2, 0) edit_button.clicked.connect(self.edit_array) self.arr = None # le tableau si il a été modifié self.instance = None self.dtype_line, self.dtype_label = get_image_layout("dtype.png", "") self.first_line.insertSpacing(2, 5) self.first_line.insertLayout(3, self.dtype_line)
[docs] def edit_array(self): """Open an array editor dialog""" parent = self.parent_layout.parent label = self.item.get_prop_value("display", "label") editor = ArrayEditor(parent) if editor.setup_and_check(self.arr, title=label): if editor.exec_(): self.update(self.arr)
[docs] def get(self): """Override AbstractDataSetWidget method""" self.arr = numpy.array(self.item.get(), copy=False) if self.item.get_prop_value("display", "transpose"): self.arr = self.arr.T self.update(self.arr)
[docs] def update(self, arr): """Override AbstractDataSetWidget method""" shape = arr.shape if len(shape) == 1: shape = (1,) + shape dim = " x ".join([str(d) for d in shape]) self.dim_label.setText(dim) format = self.item.get_prop_value("display", "format") minmax = self.item.get_prop_value("display", "minmax") real_arr = numpy.real(arr) try: if minmax == "all": mint = format % real_arr.min() maxt = format % real_arr.max() elif minmax == "columns": mint = ", ".join( [format % real_arr[r, :].min() for r in range(arr.shape[0])] ) maxt = ", ".join( [format % real_arr[r, :].max() for r in range(arr.shape[0])] ) else: mint = ", ".join( [format % real_arr[:, r].min() for r in range(arr.shape[1])] ) maxt = ", ".join( [format % real_arr[:, r].max() for r in range(arr.shape[1])] ) except (TypeError, IndexError): mint, maxt = "-", "-" self.min_label.setText(mint) self.max_label.setText(maxt) typestr = str(arr.dtype) self.dtype_label.setText("-" if typestr == "object" else typestr)
[docs] def set(self): """Override AbstractDataSetWidget method""" if self.item.get_prop_value("display", "transpose"): value = self.value().T else: value = self.value() self.item.set(value)
[docs] def value(self): return self.arr
[docs] def place_on_grid( self, layout, row, label_column, widget_column, row_span=1, column_span=1 ): """Override AbstractDataSetWidget method""" layout.addWidget(self.group, row, label_column, row_span, column_span + 1)
[docs]class ButtonWidget(AbstractDataSetWidget): """ BoolItem widget """ def __init__(self, item, parent_layout): super(ButtonWidget, self).__init__(item, parent_layout) _label = self.item.get_prop_value("display", "label") self.button = self.group = QPushButton(_label) self.button.setToolTip(item.get_help()) _icon = self.item.get_prop_value("display", "icon") if _icon is not None: if isinstance(_icon, str): _icon = get_icon(_icon) self.button.setIcon(_icon) self.button.clicked.connect(self.clicked) self.cb_value = None
[docs] def get(self): """Override AbstractDataSetWidget method""" self.cb_value = self.item.get()
[docs] def set(self): """Override AbstractDataSetWidget method""" self.item.set(self.value())
[docs] def value(self): return self.cb_value
[docs] def place_on_grid( self, layout, row, label_column, widget_column, row_span=1, column_span=1 ): """Override AbstractDataSetWidget method""" layout.addWidget(self.group, row, label_column, row_span, column_span + 1)
def clicked(self, *args): self.parent_layout.update_dataitems() callback = self.item.get_prop_value("display", "callback") self.cb_value = callback( self.item.instance, self.item.item, self.cb_value, self.button.parent() ) self.set() self.parent_layout.update_widgets()
[docs]class DataSetWidget(AbstractDataSetWidget): """ DataSet widget """ def __init__(self, item, parent_layout): super(DataSetWidget, self).__init__(item, parent_layout) self.dataset = self.klass() # Création du layout contenant les champs d'édition du signal embedded = item.get_prop_value("display", "embedded", False) if not embedded: self.group = QGroupBox(item.get_prop_value("display", "label")) else: self.group = QFrame() self.layout = QGridLayout() self.group.setLayout(self.layout) EditLayoutClass = parent_layout.__class__ self.edit = EditLayoutClass( self.parent_layout.parent, self.dataset, self.layout )
[docs] def get(self): """Override AbstractDataSetWidget method""" self.get_dataset() for widget in self.edit.widgets: widget.get()
[docs] def set(self): """Override AbstractDataSetWidget method""" for widget in self.edit.widgets: widget.set() self.set_dataset()
[docs] def get_dataset(self): """update's internal parameter representation from the item's stored value default behavior uses update_dataset and assumes internal dataset class is the same as item's value class""" item = self.item.get() update_dataset(self.dataset, item)
[docs] def set_dataset(self): """update the item's value from the internal data representation default behavior uses restore_dataset and assumes internal dataset class is the same as item's value class""" item = self.item.get() restore_dataset(self.dataset, item)
[docs] def place_on_grid( self, layout, row, label_column, widget_column, row_span=1, column_span=1 ): """Override AbstractDataSetWidget method""" layout.addWidget(self.group, row, label_column, row_span, column_span + 1)