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 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 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 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 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 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 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 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 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 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 = []
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 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 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 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)