2013-12-23 12:51:33 +00:00
|
|
|
from ..Qt import QtCore, QtGui
|
|
|
|
from ..python2_3 import asUnicode
|
2012-05-11 22:05:41 +00:00
|
|
|
from .Parameter import Parameter, registerParameterType
|
|
|
|
from .ParameterItem import ParameterItem
|
2013-12-23 12:51:33 +00:00
|
|
|
from ..widgets.SpinBox import SpinBox
|
|
|
|
from ..widgets.ColorButton import ColorButton
|
2017-05-02 00:09:15 +00:00
|
|
|
from ..colormap import ColorMap
|
2013-12-23 12:51:33 +00:00
|
|
|
#from ..widgets.GradientWidget import GradientWidget ## creates import loop
|
|
|
|
from .. import pixmaps as pixmaps
|
|
|
|
from .. import functions as fn
|
2017-05-02 00:09:15 +00:00
|
|
|
import os, sys
|
2013-12-23 12:51:33 +00:00
|
|
|
from ..pgcollections import OrderedDict
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
class WidgetParameterItem(ParameterItem):
|
|
|
|
"""
|
|
|
|
ParameterTree item with:
|
2012-08-23 15:17:40 +00:00
|
|
|
|
2013-02-10 22:45:16 +00:00
|
|
|
* label in second column for displaying value
|
|
|
|
* simple widget for editing value (displayed instead of label when item is selected)
|
|
|
|
* button that resets value to default
|
|
|
|
|
2014-02-15 10:42:36 +00:00
|
|
|
========================== =============================================================
|
|
|
|
**Registered Types:**
|
|
|
|
int Displays a :class:`SpinBox <pyqtgraph.SpinBox>` in integer
|
|
|
|
mode.
|
|
|
|
float Displays a :class:`SpinBox <pyqtgraph.SpinBox>`.
|
|
|
|
bool Displays a QCheckBox
|
|
|
|
str Displays a QLineEdit
|
|
|
|
color Displays a :class:`ColorButton <pyqtgraph.ColorButton>`
|
|
|
|
colormap Displays a :class:`GradientWidget <pyqtgraph.GradientWidget>`
|
|
|
|
========================== =============================================================
|
2012-08-23 15:17:40 +00:00
|
|
|
|
2012-03-02 02:55:32 +00:00
|
|
|
This class can be subclassed by overriding makeWidget() to provide a custom widget.
|
|
|
|
"""
|
|
|
|
def __init__(self, param, depth):
|
|
|
|
ParameterItem.__init__(self, param, depth)
|
|
|
|
|
|
|
|
self.hideWidget = True ## hide edit widget, replace with label when not selected
|
|
|
|
## set this to False to keep the editor widget always visible
|
|
|
|
|
|
|
|
|
|
|
|
## build widget into column 1 with a display label and default button.
|
|
|
|
w = self.makeWidget()
|
|
|
|
self.widget = w
|
|
|
|
self.eventProxy = EventProxy(w, self.widgetEventFilter)
|
|
|
|
|
|
|
|
opts = self.param.opts
|
|
|
|
if 'tip' in opts:
|
|
|
|
w.setToolTip(opts['tip'])
|
|
|
|
|
|
|
|
self.defaultBtn = QtGui.QPushButton()
|
|
|
|
self.defaultBtn.setFixedWidth(20)
|
|
|
|
self.defaultBtn.setFixedHeight(20)
|
|
|
|
modDir = os.path.dirname(__file__)
|
2012-10-11 15:14:35 +00:00
|
|
|
self.defaultBtn.setIcon(QtGui.QIcon(pixmaps.getPixmap('default')))
|
2012-03-02 02:55:32 +00:00
|
|
|
self.defaultBtn.clicked.connect(self.defaultClicked)
|
|
|
|
|
|
|
|
self.displayLabel = QtGui.QLabel()
|
|
|
|
|
|
|
|
layout = QtGui.QHBoxLayout()
|
|
|
|
layout.setContentsMargins(0, 0, 0, 0)
|
|
|
|
layout.setSpacing(2)
|
|
|
|
layout.addWidget(w)
|
|
|
|
layout.addWidget(self.displayLabel)
|
|
|
|
layout.addWidget(self.defaultBtn)
|
|
|
|
self.layoutWidget = QtGui.QWidget()
|
|
|
|
self.layoutWidget.setLayout(layout)
|
|
|
|
|
|
|
|
if w.sigChanged is not None:
|
|
|
|
w.sigChanged.connect(self.widgetValueChanged)
|
|
|
|
|
|
|
|
if hasattr(w, 'sigChanging'):
|
|
|
|
w.sigChanging.connect(self.widgetValueChanging)
|
|
|
|
|
|
|
|
## update value shown in widget.
|
2013-02-10 19:10:30 +00:00
|
|
|
if opts.get('value', None) is not None:
|
|
|
|
self.valueChanged(self, opts['value'], force=True)
|
|
|
|
else:
|
|
|
|
## no starting value was given; use whatever the widget has
|
|
|
|
self.widgetValueChanged()
|
2012-03-02 02:55:32 +00:00
|
|
|
|
2014-08-07 13:03:26 +00:00
|
|
|
self.updateDefaultBtn()
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
def makeWidget(self):
|
|
|
|
"""
|
|
|
|
Return a single widget that should be placed in the second tree column.
|
|
|
|
The widget must be given three attributes:
|
2012-08-23 15:17:40 +00:00
|
|
|
|
|
|
|
========== ============================================================
|
|
|
|
sigChanged a signal that is emitted when the widget's value is changed
|
|
|
|
value a function that returns the value
|
|
|
|
setValue a function that sets the value
|
|
|
|
========== ============================================================
|
|
|
|
|
2012-03-02 02:55:32 +00:00
|
|
|
This is a good function to override in subclasses.
|
|
|
|
"""
|
|
|
|
opts = self.param.opts
|
|
|
|
t = opts['type']
|
2015-06-04 02:18:02 +00:00
|
|
|
if t in ('int', 'float'):
|
2012-03-02 02:55:32 +00:00
|
|
|
defs = {
|
2015-06-04 02:18:02 +00:00
|
|
|
'value': 0, 'min': None, 'max': None,
|
2012-03-02 02:55:32 +00:00
|
|
|
'step': 1.0, 'dec': False,
|
2015-06-04 02:18:02 +00:00
|
|
|
'siPrefix': False, 'suffix': '', 'decimals': 3,
|
2012-03-02 02:55:32 +00:00
|
|
|
}
|
2015-06-04 02:18:02 +00:00
|
|
|
if t == 'int':
|
|
|
|
defs['int'] = True
|
|
|
|
defs['minStep'] = 1.0
|
2018-04-25 21:54:35 +00:00
|
|
|
defs['format'] = '{value:d}'
|
2015-06-04 02:18:02 +00:00
|
|
|
for k in defs:
|
|
|
|
if k in opts:
|
|
|
|
defs[k] = opts[k]
|
2012-03-02 02:55:32 +00:00
|
|
|
if 'limits' in opts:
|
2016-05-03 18:25:05 +00:00
|
|
|
defs['min'], defs['max'] = opts['limits']
|
2012-03-02 02:55:32 +00:00
|
|
|
w = SpinBox()
|
|
|
|
w.setOpts(**defs)
|
|
|
|
w.sigChanged = w.sigValueChanged
|
|
|
|
w.sigChanging = w.sigValueChanging
|
|
|
|
elif t == 'bool':
|
|
|
|
w = QtGui.QCheckBox()
|
|
|
|
w.sigChanged = w.toggled
|
|
|
|
w.value = w.isChecked
|
|
|
|
w.setValue = w.setChecked
|
2014-05-04 16:24:46 +00:00
|
|
|
w.setEnabled(not opts.get('readonly', False))
|
2012-03-02 02:55:32 +00:00
|
|
|
self.hideWidget = False
|
|
|
|
elif t == 'str':
|
|
|
|
w = QtGui.QLineEdit()
|
2016-12-08 01:34:44 +00:00
|
|
|
w.setStyleSheet('border: 0px')
|
2012-03-02 02:55:32 +00:00
|
|
|
w.sigChanged = w.editingFinished
|
2012-05-11 22:05:41 +00:00
|
|
|
w.value = lambda: asUnicode(w.text())
|
|
|
|
w.setValue = lambda v: w.setText(asUnicode(v))
|
2012-03-02 02:55:32 +00:00
|
|
|
w.sigChanging = w.textChanged
|
|
|
|
elif t == 'color':
|
|
|
|
w = ColorButton()
|
|
|
|
w.sigChanged = w.sigColorChanged
|
|
|
|
w.sigChanging = w.sigColorChanging
|
|
|
|
w.value = w.color
|
|
|
|
w.setValue = w.setColor
|
|
|
|
self.hideWidget = False
|
|
|
|
w.setFlat(True)
|
2014-05-04 16:24:46 +00:00
|
|
|
w.setEnabled(not opts.get('readonly', False))
|
2013-02-10 19:10:30 +00:00
|
|
|
elif t == 'colormap':
|
2013-12-23 12:51:33 +00:00
|
|
|
from ..widgets.GradientWidget import GradientWidget ## need this here to avoid import loop
|
2013-02-10 19:10:30 +00:00
|
|
|
w = GradientWidget(orientation='bottom')
|
|
|
|
w.sigChanged = w.sigGradientChangeFinished
|
|
|
|
w.sigChanging = w.sigGradientChanged
|
|
|
|
w.value = w.colorMap
|
|
|
|
w.setValue = w.setColorMap
|
|
|
|
self.hideWidget = False
|
2012-03-02 02:55:32 +00:00
|
|
|
else:
|
2012-05-11 22:05:41 +00:00
|
|
|
raise Exception("Unknown type '%s'" % asUnicode(t))
|
2012-03-02 02:55:32 +00:00
|
|
|
return w
|
|
|
|
|
|
|
|
def widgetEventFilter(self, obj, ev):
|
|
|
|
## filter widget's events
|
|
|
|
## catch TAB to change focus
|
|
|
|
## catch focusOut to hide editor
|
|
|
|
if ev.type() == ev.KeyPress:
|
|
|
|
if ev.key() == QtCore.Qt.Key_Tab:
|
|
|
|
self.focusNext(forward=True)
|
|
|
|
return True ## don't let anyone else see this event
|
|
|
|
elif ev.key() == QtCore.Qt.Key_Backtab:
|
|
|
|
self.focusNext(forward=False)
|
|
|
|
return True ## don't let anyone else see this event
|
|
|
|
|
|
|
|
#elif ev.type() == ev.FocusOut:
|
|
|
|
#self.hideEditor()
|
|
|
|
return False
|
|
|
|
|
|
|
|
def setFocus(self):
|
|
|
|
self.showEditor()
|
|
|
|
|
|
|
|
def isFocusable(self):
|
|
|
|
return self.param.writable()
|
|
|
|
|
|
|
|
def valueChanged(self, param, val, force=False):
|
|
|
|
## called when the parameter's value has changed
|
|
|
|
ParameterItem.valueChanged(self, param, val)
|
|
|
|
self.widget.sigChanged.disconnect(self.widgetValueChanged)
|
|
|
|
try:
|
|
|
|
if force or val != self.widget.value():
|
|
|
|
self.widget.setValue(val)
|
|
|
|
self.updateDisplayLabel(val) ## always make sure label is updated, even if values match!
|
|
|
|
finally:
|
|
|
|
self.widget.sigChanged.connect(self.widgetValueChanged)
|
|
|
|
self.updateDefaultBtn()
|
|
|
|
|
|
|
|
def updateDefaultBtn(self):
|
|
|
|
## enable/disable default btn
|
|
|
|
self.defaultBtn.setEnabled(not self.param.valueIsDefault() and self.param.writable())
|
2014-08-07 13:03:26 +00:00
|
|
|
|
|
|
|
# hide / show
|
|
|
|
self.defaultBtn.setVisible(not self.param.readonly())
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
def updateDisplayLabel(self, value=None):
|
|
|
|
"""Update the display label to reflect the value of the parameter."""
|
|
|
|
if value is None:
|
|
|
|
value = self.param.value()
|
|
|
|
opts = self.param.opts
|
|
|
|
if isinstance(self.widget, QtGui.QAbstractSpinBox):
|
2012-05-11 22:05:41 +00:00
|
|
|
text = asUnicode(self.widget.lineEdit().text())
|
2012-03-02 02:55:32 +00:00
|
|
|
elif isinstance(self.widget, QtGui.QComboBox):
|
|
|
|
text = self.widget.currentText()
|
|
|
|
else:
|
2012-05-11 22:05:41 +00:00
|
|
|
text = asUnicode(value)
|
2012-03-02 02:55:32 +00:00
|
|
|
self.displayLabel.setText(text)
|
|
|
|
|
|
|
|
def widgetValueChanged(self):
|
|
|
|
## called when the widget's value has been changed by the user
|
|
|
|
val = self.widget.value()
|
|
|
|
newVal = self.param.setValue(val)
|
|
|
|
|
2014-03-25 17:15:29 +00:00
|
|
|
def widgetValueChanging(self, *args):
|
2012-03-02 02:55:32 +00:00
|
|
|
"""
|
|
|
|
Called when the widget's value is changing, but not finalized.
|
|
|
|
For example: editing text before pressing enter or changing focus.
|
|
|
|
"""
|
2014-03-25 17:15:29 +00:00
|
|
|
# This is a bit sketchy: assume the last argument of each signal is
|
|
|
|
# the value..
|
|
|
|
self.param.sigValueChanging.emit(self.param, args[-1])
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
def selected(self, sel):
|
|
|
|
"""Called when this item has been selected (sel=True) OR deselected (sel=False)"""
|
|
|
|
ParameterItem.selected(self, sel)
|
|
|
|
|
|
|
|
if self.widget is None:
|
|
|
|
return
|
|
|
|
if sel and self.param.writable():
|
|
|
|
self.showEditor()
|
|
|
|
elif self.hideWidget:
|
|
|
|
self.hideEditor()
|
|
|
|
|
|
|
|
def showEditor(self):
|
|
|
|
self.widget.show()
|
|
|
|
self.displayLabel.hide()
|
|
|
|
self.widget.setFocus(QtCore.Qt.OtherFocusReason)
|
2014-08-07 13:03:26 +00:00
|
|
|
if isinstance(self.widget, SpinBox):
|
|
|
|
self.widget.selectNumber() # select the numerical portion of the text for quick editing
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
def hideEditor(self):
|
|
|
|
self.widget.hide()
|
|
|
|
self.displayLabel.show()
|
|
|
|
|
|
|
|
def limitsChanged(self, param, limits):
|
|
|
|
"""Called when the parameter's limits have changed"""
|
|
|
|
ParameterItem.limitsChanged(self, param, limits)
|
|
|
|
|
|
|
|
t = self.param.opts['type']
|
|
|
|
if t == 'int' or t == 'float':
|
|
|
|
self.widget.setOpts(bounds=limits)
|
|
|
|
else:
|
|
|
|
return ## don't know what to do with any other types..
|
|
|
|
|
|
|
|
def defaultChanged(self, param, value):
|
|
|
|
self.updateDefaultBtn()
|
|
|
|
|
|
|
|
def treeWidgetChanged(self):
|
|
|
|
"""Called when this item is added or removed from a tree."""
|
|
|
|
ParameterItem.treeWidgetChanged(self)
|
|
|
|
|
|
|
|
## add all widgets for this item into the tree
|
|
|
|
if self.widget is not None:
|
|
|
|
tree = self.treeWidget()
|
|
|
|
if tree is None:
|
|
|
|
return
|
|
|
|
tree.setItemWidget(self, 1, self.layoutWidget)
|
|
|
|
self.displayLabel.hide()
|
|
|
|
self.selected(False)
|
|
|
|
|
|
|
|
def defaultClicked(self):
|
|
|
|
self.param.setToDefault()
|
|
|
|
|
|
|
|
def optsChanged(self, param, opts):
|
|
|
|
"""Called when any options are changed that are not
|
|
|
|
name, value, default, or limits"""
|
|
|
|
#print "opts changed:", opts
|
|
|
|
ParameterItem.optsChanged(self, param, opts)
|
|
|
|
|
|
|
|
if 'readonly' in opts:
|
|
|
|
self.updateDefaultBtn()
|
2014-05-04 16:24:46 +00:00
|
|
|
if isinstance(self.widget, (QtGui.QCheckBox,ColorButton)):
|
2014-08-07 13:03:26 +00:00
|
|
|
self.widget.setEnabled(not opts['readonly'])
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
## If widget is a SpinBox, pass options straight through
|
|
|
|
if isinstance(self.widget, SpinBox):
|
2016-12-07 06:56:55 +00:00
|
|
|
# send only options supported by spinbox
|
|
|
|
sbOpts = {}
|
2012-03-02 02:55:32 +00:00
|
|
|
if 'units' in opts and 'suffix' not in opts:
|
2016-12-07 06:56:55 +00:00
|
|
|
sbOpts['suffix'] = opts['units']
|
|
|
|
for k,v in opts.items():
|
|
|
|
if k in self.widget.opts:
|
|
|
|
sbOpts[k] = v
|
|
|
|
self.widget.setOpts(**sbOpts)
|
2012-03-02 02:55:32 +00:00
|
|
|
self.updateDisplayLabel()
|
2014-05-04 16:24:46 +00:00
|
|
|
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
class EventProxy(QtCore.QObject):
|
|
|
|
def __init__(self, qobj, callback):
|
|
|
|
QtCore.QObject.__init__(self)
|
|
|
|
self.callback = callback
|
|
|
|
qobj.installEventFilter(self)
|
|
|
|
|
|
|
|
def eventFilter(self, obj, ev):
|
|
|
|
return self.callback(obj, ev)
|
|
|
|
|
|
|
|
|
|
|
|
class SimpleParameter(Parameter):
|
|
|
|
itemClass = WidgetParameterItem
|
|
|
|
|
2012-03-11 15:59:45 +00:00
|
|
|
def __init__(self, *args, **kargs):
|
|
|
|
Parameter.__init__(self, *args, **kargs)
|
2012-09-09 23:13:30 +00:00
|
|
|
|
|
|
|
## override a few methods for color parameters
|
2012-03-11 15:59:45 +00:00
|
|
|
if self.opts['type'] == 'color':
|
|
|
|
self.value = self.colorValue
|
2012-09-09 23:13:30 +00:00
|
|
|
self.saveState = self.saveColorState
|
2012-03-11 15:59:45 +00:00
|
|
|
|
|
|
|
def colorValue(self):
|
2013-12-23 12:51:33 +00:00
|
|
|
return fn.mkColor(Parameter.value(self))
|
2012-03-11 15:59:45 +00:00
|
|
|
|
2014-08-07 13:03:26 +00:00
|
|
|
def saveColorState(self, *args, **kwds):
|
|
|
|
state = Parameter.saveState(self, *args, **kwds)
|
2013-12-23 12:51:33 +00:00
|
|
|
state['value'] = fn.colorTuple(self.value())
|
2012-09-09 23:13:30 +00:00
|
|
|
return state
|
|
|
|
|
2017-05-02 00:09:15 +00:00
|
|
|
def _interpretValue(self, v):
|
|
|
|
fn = {
|
|
|
|
'int': int,
|
|
|
|
'float': float,
|
|
|
|
'bool': bool,
|
2017-08-01 00:16:46 +00:00
|
|
|
'str': asUnicode,
|
2017-05-02 00:09:15 +00:00
|
|
|
'color': self._interpColor,
|
|
|
|
'colormap': self._interpColormap,
|
|
|
|
}[self.opts['type']]
|
|
|
|
return fn(v)
|
|
|
|
|
|
|
|
def _interpColor(self, v):
|
|
|
|
return fn.mkColor(v)
|
|
|
|
|
|
|
|
def _interpColormap(self, v):
|
|
|
|
if not isinstance(v, ColorMap):
|
|
|
|
raise TypeError("Cannot set colormap parameter from object %r" % v)
|
|
|
|
return v
|
|
|
|
|
|
|
|
|
2012-09-09 23:13:30 +00:00
|
|
|
|
2012-03-02 02:55:32 +00:00
|
|
|
registerParameterType('int', SimpleParameter, override=True)
|
|
|
|
registerParameterType('float', SimpleParameter, override=True)
|
|
|
|
registerParameterType('bool', SimpleParameter, override=True)
|
|
|
|
registerParameterType('str', SimpleParameter, override=True)
|
|
|
|
registerParameterType('color', SimpleParameter, override=True)
|
2013-02-10 19:10:30 +00:00
|
|
|
registerParameterType('colormap', SimpleParameter, override=True)
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class GroupParameterItem(ParameterItem):
|
|
|
|
"""
|
|
|
|
Group parameters are used mainly as a generic parent item that holds (and groups!) a set
|
|
|
|
of child parameters. It also provides a simple mechanism for displaying a button or combo
|
|
|
|
that can be used to add new parameters to the group.
|
|
|
|
"""
|
|
|
|
def __init__(self, param, depth):
|
|
|
|
ParameterItem.__init__(self, param, depth)
|
2012-08-31 21:18:06 +00:00
|
|
|
self.updateDepth(depth)
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
self.addItem = None
|
|
|
|
if 'addText' in param.opts:
|
|
|
|
addText = param.opts['addText']
|
|
|
|
if 'addList' in param.opts:
|
|
|
|
self.addWidget = QtGui.QComboBox()
|
2012-09-09 23:13:30 +00:00
|
|
|
self.addWidget.setSizeAdjustPolicy(QtGui.QComboBox.AdjustToContents)
|
|
|
|
self.updateAddList()
|
2012-03-02 02:55:32 +00:00
|
|
|
self.addWidget.currentIndexChanged.connect(self.addChanged)
|
|
|
|
else:
|
|
|
|
self.addWidget = QtGui.QPushButton(addText)
|
|
|
|
self.addWidget.clicked.connect(self.addClicked)
|
|
|
|
w = QtGui.QWidget()
|
|
|
|
l = QtGui.QHBoxLayout()
|
|
|
|
l.setContentsMargins(0,0,0,0)
|
|
|
|
w.setLayout(l)
|
|
|
|
l.addWidget(self.addWidget)
|
2012-09-09 23:13:30 +00:00
|
|
|
l.addStretch()
|
|
|
|
#l.addItem(QtGui.QSpacerItem(200, 10, QtGui.QSizePolicy.Expanding, QtGui.QSizePolicy.Minimum))
|
2012-03-02 02:55:32 +00:00
|
|
|
self.addWidgetBox = w
|
|
|
|
self.addItem = QtGui.QTreeWidgetItem([])
|
|
|
|
self.addItem.setFlags(QtCore.Qt.ItemIsEnabled)
|
|
|
|
ParameterItem.addChild(self, self.addItem)
|
|
|
|
|
2012-08-31 21:18:06 +00:00
|
|
|
def updateDepth(self, depth):
|
|
|
|
## Change item's appearance based on its depth in the tree
|
|
|
|
## This allows highest-level groups to be displayed more prominently.
|
|
|
|
if depth == 0:
|
|
|
|
for c in [0,1]:
|
|
|
|
self.setBackground(c, QtGui.QBrush(QtGui.QColor(100,100,100)))
|
|
|
|
self.setForeground(c, QtGui.QBrush(QtGui.QColor(220,220,255)))
|
|
|
|
font = self.font(c)
|
|
|
|
font.setBold(True)
|
|
|
|
font.setPointSize(font.pointSize()+1)
|
|
|
|
self.setFont(c, font)
|
|
|
|
self.setSizeHint(0, QtCore.QSize(0, 25))
|
|
|
|
else:
|
|
|
|
for c in [0,1]:
|
|
|
|
self.setBackground(c, QtGui.QBrush(QtGui.QColor(220,220,220)))
|
2018-01-31 16:44:09 +00:00
|
|
|
self.setForeground(c, QtGui.QBrush(QtGui.QColor(50,50,50)))
|
2012-08-31 21:18:06 +00:00
|
|
|
font = self.font(c)
|
|
|
|
font.setBold(True)
|
|
|
|
#font.setPointSize(font.pointSize()+1)
|
|
|
|
self.setFont(c, font)
|
|
|
|
self.setSizeHint(0, QtCore.QSize(0, 20))
|
|
|
|
|
2012-03-02 02:55:32 +00:00
|
|
|
def addClicked(self):
|
|
|
|
"""Called when "add new" button is clicked
|
|
|
|
The parameter MUST have an 'addNew' method defined.
|
|
|
|
"""
|
|
|
|
self.param.addNew()
|
|
|
|
|
|
|
|
def addChanged(self):
|
|
|
|
"""Called when "add new" combo is changed
|
|
|
|
The parameter MUST have an 'addNew' method defined.
|
|
|
|
"""
|
|
|
|
if self.addWidget.currentIndex() == 0:
|
|
|
|
return
|
2012-05-11 22:05:41 +00:00
|
|
|
typ = asUnicode(self.addWidget.currentText())
|
2012-03-02 02:55:32 +00:00
|
|
|
self.param.addNew(typ)
|
|
|
|
self.addWidget.setCurrentIndex(0)
|
|
|
|
|
|
|
|
def treeWidgetChanged(self):
|
|
|
|
ParameterItem.treeWidgetChanged(self)
|
2012-07-09 21:14:41 +00:00
|
|
|
self.treeWidget().setFirstItemColumnSpanned(self, True)
|
2012-03-02 02:55:32 +00:00
|
|
|
if self.addItem is not None:
|
|
|
|
self.treeWidget().setItemWidget(self.addItem, 0, self.addWidgetBox)
|
|
|
|
self.treeWidget().setFirstItemColumnSpanned(self.addItem, True)
|
|
|
|
|
|
|
|
def addChild(self, child): ## make sure added childs are actually inserted before add btn
|
|
|
|
if self.addItem is not None:
|
|
|
|
ParameterItem.insertChild(self, self.childCount()-1, child)
|
|
|
|
else:
|
|
|
|
ParameterItem.addChild(self, child)
|
2012-09-09 23:13:30 +00:00
|
|
|
|
|
|
|
def optsChanged(self, param, changed):
|
|
|
|
if 'addList' in changed:
|
|
|
|
self.updateAddList()
|
|
|
|
|
|
|
|
def updateAddList(self):
|
|
|
|
self.addWidget.blockSignals(True)
|
|
|
|
try:
|
|
|
|
self.addWidget.clear()
|
|
|
|
self.addWidget.addItem(self.param.opts['addText'])
|
|
|
|
for t in self.param.opts['addList']:
|
|
|
|
self.addWidget.addItem(t)
|
|
|
|
finally:
|
|
|
|
self.addWidget.blockSignals(False)
|
|
|
|
|
2012-03-02 02:55:32 +00:00
|
|
|
class GroupParameter(Parameter):
|
|
|
|
"""
|
|
|
|
Group parameters are used mainly as a generic parent item that holds (and groups!) a set
|
2012-09-09 23:13:30 +00:00
|
|
|
of child parameters.
|
|
|
|
|
|
|
|
It also provides a simple mechanism for displaying a button or combo
|
|
|
|
that can be used to add new parameters to the group. To enable this, the group
|
|
|
|
must be initialized with the 'addText' option (the text will be displayed on
|
|
|
|
a button which, when clicked, will cause addNew() to be called). If the 'addList'
|
|
|
|
option is specified as well, then a dropdown-list of addable items will be displayed
|
|
|
|
instead of a button.
|
2012-03-02 02:55:32 +00:00
|
|
|
"""
|
|
|
|
itemClass = GroupParameterItem
|
2017-09-29 15:59:14 +00:00
|
|
|
|
|
|
|
sigAddNew = QtCore.Signal(object, object) # self, type
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
def addNew(self, typ=None):
|
2012-09-09 23:13:30 +00:00
|
|
|
"""
|
|
|
|
This method is called when the user has requested to add a new item to the group.
|
2017-09-29 15:59:14 +00:00
|
|
|
By default, it emits ``sigAddNew(self, typ)``.
|
2012-09-09 23:13:30 +00:00
|
|
|
"""
|
2017-09-29 15:59:14 +00:00
|
|
|
self.sigAddNew.emit(self, typ)
|
2012-09-09 23:13:30 +00:00
|
|
|
|
|
|
|
def setAddList(self, vals):
|
|
|
|
"""Change the list of options available for the user to add to the group."""
|
|
|
|
self.setOpts(addList=vals)
|
|
|
|
|
|
|
|
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
registerParameterType('group', GroupParameter, override=True)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ListParameterItem(WidgetParameterItem):
|
|
|
|
"""
|
|
|
|
WidgetParameterItem subclass providing comboBox that lets the user select from a list of options.
|
|
|
|
|
|
|
|
"""
|
|
|
|
def __init__(self, param, depth):
|
2012-09-09 23:13:30 +00:00
|
|
|
self.targetValue = None
|
2012-03-02 02:55:32 +00:00
|
|
|
WidgetParameterItem.__init__(self, param, depth)
|
|
|
|
|
2012-09-09 23:13:30 +00:00
|
|
|
|
2012-03-02 02:55:32 +00:00
|
|
|
def makeWidget(self):
|
|
|
|
opts = self.param.opts
|
|
|
|
t = opts['type']
|
|
|
|
w = QtGui.QComboBox()
|
|
|
|
w.setMaximumHeight(20) ## set to match height of spin box and line edit
|
|
|
|
w.sigChanged = w.currentIndexChanged
|
|
|
|
w.value = self.value
|
|
|
|
w.setValue = self.setValue
|
|
|
|
self.widget = w ## needs to be set before limits are changed
|
|
|
|
self.limitsChanged(self.param, self.param.opts['limits'])
|
|
|
|
if len(self.forward) > 0:
|
|
|
|
self.setValue(self.param.value())
|
|
|
|
return w
|
|
|
|
|
|
|
|
def value(self):
|
2012-05-11 22:05:41 +00:00
|
|
|
key = asUnicode(self.widget.currentText())
|
2012-09-09 23:13:30 +00:00
|
|
|
|
|
|
|
return self.forward.get(key, None)
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
def setValue(self, val):
|
2012-09-09 23:13:30 +00:00
|
|
|
self.targetValue = val
|
2013-07-12 17:14:09 +00:00
|
|
|
if val not in self.reverse[0]:
|
2012-03-02 02:55:32 +00:00
|
|
|
self.widget.setCurrentIndex(0)
|
|
|
|
else:
|
2013-07-12 17:14:09 +00:00
|
|
|
key = self.reverse[1][self.reverse[0].index(val)]
|
2012-03-02 02:55:32 +00:00
|
|
|
ind = self.widget.findText(key)
|
|
|
|
self.widget.setCurrentIndex(ind)
|
|
|
|
|
|
|
|
def limitsChanged(self, param, limits):
|
|
|
|
# set up forward / reverse mappings for name:value
|
|
|
|
|
2012-09-09 23:13:30 +00:00
|
|
|
if len(limits) == 0:
|
|
|
|
limits = [''] ## Can never have an empty list--there is always at least a singhe blank item.
|
|
|
|
|
|
|
|
self.forward, self.reverse = ListParameter.mapping(limits)
|
2012-03-02 02:55:32 +00:00
|
|
|
try:
|
|
|
|
self.widget.blockSignals(True)
|
2012-09-09 23:13:30 +00:00
|
|
|
val = self.targetValue #asUnicode(self.widget.currentText())
|
|
|
|
|
2012-03-02 02:55:32 +00:00
|
|
|
self.widget.clear()
|
|
|
|
for k in self.forward:
|
|
|
|
self.widget.addItem(k)
|
|
|
|
if k == val:
|
|
|
|
self.widget.setCurrentIndex(self.widget.count()-1)
|
2012-09-09 23:13:30 +00:00
|
|
|
self.updateDisplayLabel()
|
2012-03-02 02:55:32 +00:00
|
|
|
finally:
|
|
|
|
self.widget.blockSignals(False)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ListParameter(Parameter):
|
|
|
|
itemClass = ListParameterItem
|
|
|
|
|
|
|
|
def __init__(self, **opts):
|
2013-07-12 17:14:09 +00:00
|
|
|
self.forward = OrderedDict() ## {name: value, ...}
|
|
|
|
self.reverse = ([], []) ## ([value, ...], [name, ...])
|
2012-09-09 23:13:30 +00:00
|
|
|
|
|
|
|
## Parameter uses 'limits' option to define the set of allowed values
|
2012-03-02 02:55:32 +00:00
|
|
|
if 'values' in opts:
|
|
|
|
opts['limits'] = opts['values']
|
2012-09-09 23:13:30 +00:00
|
|
|
if opts.get('limits', None) is None:
|
|
|
|
opts['limits'] = []
|
2012-03-02 02:55:32 +00:00
|
|
|
Parameter.__init__(self, **opts)
|
2012-10-31 05:53:16 +00:00
|
|
|
self.setLimits(opts['limits'])
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
def setLimits(self, limits):
|
2012-09-09 23:13:30 +00:00
|
|
|
self.forward, self.reverse = self.mapping(limits)
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
Parameter.setLimits(self, limits)
|
2014-05-22 03:25:07 +00:00
|
|
|
if len(self.reverse[0]) > 0 and self.value() not in self.reverse[0]:
|
2013-07-12 17:14:09 +00:00
|
|
|
self.setValue(self.reverse[0][0])
|
|
|
|
|
|
|
|
#def addItem(self, name, value=None):
|
|
|
|
#if name in self.forward:
|
|
|
|
#raise Exception("Name '%s' is already in use for this parameter" % name)
|
|
|
|
#limits = self.opts['limits']
|
|
|
|
#if isinstance(limits, dict):
|
|
|
|
#limits = limits.copy()
|
|
|
|
#limits[name] = value
|
|
|
|
#self.setLimits(limits)
|
|
|
|
#else:
|
|
|
|
#if value is not None:
|
|
|
|
#raise Exception ## raise exception or convert to dict?
|
|
|
|
#limits = limits[:]
|
|
|
|
#limits.append(name)
|
|
|
|
## what if limits == None?
|
2012-03-02 02:55:32 +00:00
|
|
|
|
2012-09-09 23:13:30 +00:00
|
|
|
@staticmethod
|
|
|
|
def mapping(limits):
|
2013-07-12 17:14:09 +00:00
|
|
|
## Return forward and reverse mapping objects given a limit specification
|
|
|
|
forward = OrderedDict() ## {name: value, ...}
|
|
|
|
reverse = ([], []) ## ([value, ...], [name, ...])
|
2012-09-09 23:13:30 +00:00
|
|
|
if isinstance(limits, dict):
|
|
|
|
for k, v in limits.items():
|
|
|
|
forward[k] = v
|
2013-07-12 17:14:09 +00:00
|
|
|
reverse[0].append(v)
|
|
|
|
reverse[1].append(k)
|
2012-09-09 23:13:30 +00:00
|
|
|
else:
|
|
|
|
for v in limits:
|
|
|
|
n = asUnicode(v)
|
|
|
|
forward[n] = v
|
2013-07-12 17:14:09 +00:00
|
|
|
reverse[0].append(v)
|
|
|
|
reverse[1].append(n)
|
2012-09-09 23:13:30 +00:00
|
|
|
return forward, reverse
|
2012-03-02 02:55:32 +00:00
|
|
|
|
|
|
|
registerParameterType('list', ListParameter, override=True)
|
|
|
|
|
|
|
|
|
2012-09-09 23:13:30 +00:00
|
|
|
|
|
|
|
class ActionParameterItem(ParameterItem):
|
|
|
|
def __init__(self, param, depth):
|
|
|
|
ParameterItem.__init__(self, param, depth)
|
|
|
|
self.layoutWidget = QtGui.QWidget()
|
|
|
|
self.layout = QtGui.QHBoxLayout()
|
2017-09-29 15:59:37 +00:00
|
|
|
self.layout.setContentsMargins(0, 0, 0, 0)
|
2012-09-09 23:13:30 +00:00
|
|
|
self.layoutWidget.setLayout(self.layout)
|
|
|
|
self.button = QtGui.QPushButton(param.name())
|
|
|
|
#self.layout.addSpacing(100)
|
|
|
|
self.layout.addWidget(self.button)
|
|
|
|
self.layout.addStretch()
|
|
|
|
self.button.clicked.connect(self.buttonClicked)
|
|
|
|
param.sigNameChanged.connect(self.paramRenamed)
|
|
|
|
self.setText(0, '')
|
|
|
|
|
|
|
|
def treeWidgetChanged(self):
|
|
|
|
ParameterItem.treeWidgetChanged(self)
|
|
|
|
tree = self.treeWidget()
|
|
|
|
if tree is None:
|
|
|
|
return
|
|
|
|
|
|
|
|
tree.setFirstItemColumnSpanned(self, True)
|
|
|
|
tree.setItemWidget(self, 0, self.layoutWidget)
|
|
|
|
|
|
|
|
def paramRenamed(self, param, name):
|
|
|
|
self.button.setText(name)
|
|
|
|
|
|
|
|
def buttonClicked(self):
|
|
|
|
self.param.activate()
|
|
|
|
|
|
|
|
class ActionParameter(Parameter):
|
|
|
|
"""Used for displaying a button within the tree."""
|
|
|
|
itemClass = ActionParameterItem
|
|
|
|
sigActivated = QtCore.Signal(object)
|
|
|
|
|
|
|
|
def activate(self):
|
|
|
|
self.sigActivated.emit(self)
|
|
|
|
self.emitStateChanged('activated', None)
|
|
|
|
|
|
|
|
registerParameterType('action', ActionParameter, override=True)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class TextParameterItem(WidgetParameterItem):
|
|
|
|
def __init__(self, param, depth):
|
|
|
|
WidgetParameterItem.__init__(self, param, depth)
|
2013-08-27 18:00:26 +00:00
|
|
|
self.hideWidget = False
|
2012-09-09 23:13:30 +00:00
|
|
|
self.subItem = QtGui.QTreeWidgetItem()
|
|
|
|
self.addChild(self.subItem)
|
|
|
|
|
|
|
|
def treeWidgetChanged(self):
|
2013-07-10 04:02:16 +00:00
|
|
|
## TODO: fix so that superclass method can be called
|
|
|
|
## (WidgetParameter should just natively support this style)
|
|
|
|
#WidgetParameterItem.treeWidgetChanged(self)
|
2012-09-09 23:13:30 +00:00
|
|
|
self.treeWidget().setFirstItemColumnSpanned(self.subItem, True)
|
|
|
|
self.treeWidget().setItemWidget(self.subItem, 0, self.textBox)
|
2013-07-10 04:02:16 +00:00
|
|
|
|
|
|
|
# for now, these are copied from ParameterItem.treeWidgetChanged
|
|
|
|
self.setHidden(not self.param.opts.get('visible', True))
|
|
|
|
self.setExpanded(self.param.opts.get('expanded', True))
|
2012-09-09 23:13:30 +00:00
|
|
|
|
|
|
|
def makeWidget(self):
|
|
|
|
self.textBox = QtGui.QTextEdit()
|
|
|
|
self.textBox.setMaximumHeight(100)
|
2014-04-30 17:00:56 +00:00
|
|
|
self.textBox.setReadOnly(self.param.opts.get('readonly', False))
|
2012-09-09 23:13:30 +00:00
|
|
|
self.textBox.value = lambda: str(self.textBox.toPlainText())
|
|
|
|
self.textBox.setValue = self.textBox.setPlainText
|
|
|
|
self.textBox.sigChanged = self.textBox.textChanged
|
|
|
|
return self.textBox
|
|
|
|
|
|
|
|
class TextParameter(Parameter):
|
|
|
|
"""Editable string; displayed as large text box in the tree."""
|
|
|
|
itemClass = TextParameterItem
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
registerParameterType('text', TextParameter, override=True)
|