123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216 |
- import re
- from PyQt5 import QtCore
- from PyQt5 import QtGui
- from PyQt5 import QtWidgets
- from PyQt5.Qt import QT_VERSION_STR
- from libs.utils import newIcon, labelValidator
- QT5 = QT_VERSION_STR[0] == '5'
- # TODO(unknown):
- # - Calculate optimal position so as not to go out of screen area.
- class KeyQLineEdit(QtWidgets.QLineEdit):
- def setListWidget(self, list_widget):
- self.list_widget = list_widget
- def keyPressEvent(self, e):
- if e.key() in [QtCore.Qt.Key_Up, QtCore.Qt.Key_Down]:
- self.list_widget.keyPressEvent(e)
- else:
- super(KeyQLineEdit, self).keyPressEvent(e)
- class KeyDialog(QtWidgets.QDialog):
- def __init__(
- self,
- text="Enter object label",
- parent=None,
- labels=None,
- sort_labels=True,
- show_text_field=True,
- completion="startswith",
- fit_to_content=None,
- flags=None,
- ):
- if fit_to_content is None:
- fit_to_content = {"row": False, "column": True}
- self._fit_to_content = fit_to_content
- super(KeyDialog, self).__init__(parent)
- self.edit = KeyQLineEdit()
- self.edit.setPlaceholderText(text)
- self.edit.setValidator(labelValidator())
- self.edit.editingFinished.connect(self.postProcess)
- if flags:
- self.edit.textChanged.connect(self.updateFlags)
- layout = QtWidgets.QVBoxLayout()
- if show_text_field:
- layout_edit = QtWidgets.QHBoxLayout()
- layout_edit.addWidget(self.edit, 6)
- layout.addLayout(layout_edit)
- # buttons
- self.buttonBox = bb = QtWidgets.QDialogButtonBox(
- QtWidgets.QDialogButtonBox.Ok | QtWidgets.QDialogButtonBox.Cancel,
- QtCore.Qt.Horizontal,
- self,
- )
- bb.button(bb.Ok).setIcon(newIcon("done"))
- bb.button(bb.Cancel).setIcon(newIcon("undo"))
- bb.accepted.connect(self.validate)
- bb.rejected.connect(self.reject)
- layout.addWidget(bb)
- # label_list
- self.labelList = QtWidgets.QListWidget()
- if self._fit_to_content["row"]:
- self.labelList.setHorizontalScrollBarPolicy(
- QtCore.Qt.ScrollBarAlwaysOff
- )
- if self._fit_to_content["column"]:
- self.labelList.setVerticalScrollBarPolicy(
- QtCore.Qt.ScrollBarAlwaysOff
- )
- self._sort_labels = sort_labels
- if labels:
- self.labelList.addItems(labels)
- if self._sort_labels:
- self.labelList.sortItems()
- else:
- self.labelList.setDragDropMode(
- QtWidgets.QAbstractItemView.InternalMove
- )
- self.labelList.currentItemChanged.connect(self.labelSelected)
- self.labelList.itemDoubleClicked.connect(self.labelDoubleClicked)
- self.edit.setListWidget(self.labelList)
- layout.addWidget(self.labelList)
- # label_flags
- if flags is None:
- flags = {}
- self._flags = flags
- self.flagsLayout = QtWidgets.QVBoxLayout()
- self.resetFlags()
- layout.addItem(self.flagsLayout)
- self.edit.textChanged.connect(self.updateFlags)
- self.setLayout(layout)
- # completion
- completer = QtWidgets.QCompleter()
- if not QT5 and completion != "startswith":
- completion = "startswith"
- if completion == "startswith":
- completer.setCompletionMode(QtWidgets.QCompleter.InlineCompletion)
- # Default settings.
- # completer.setFilterMode(QtCore.Qt.MatchStartsWith)
- elif completion == "contains":
- completer.setCompletionMode(QtWidgets.QCompleter.PopupCompletion)
- completer.setFilterMode(QtCore.Qt.MatchContains)
- else:
- raise ValueError("Unsupported completion: {}".format(completion))
- completer.setModel(self.labelList.model())
- self.edit.setCompleter(completer)
- def addLabelHistory(self, label):
- if self.labelList.findItems(label, QtCore.Qt.MatchExactly):
- return
- self.labelList.addItem(label)
- if self._sort_labels:
- self.labelList.sortItems()
- def labelSelected(self, item):
- self.edit.setText(item.text())
- def validate(self):
- text = self.edit.text()
- if hasattr(text, "strip"):
- text = text.strip()
- else:
- text = text.trimmed()
- if text:
- self.accept()
- def labelDoubleClicked(self, item):
- self.validate()
- def postProcess(self):
- text = self.edit.text()
- if hasattr(text, "strip"):
- text = text.strip()
- else:
- text = text.trimmed()
- self.edit.setText(text)
- def updateFlags(self, label_new):
- # keep state of shared flags
- flags_old = self.getFlags()
- flags_new = {}
- for pattern, keys in self._flags.items():
- if re.match(pattern, label_new):
- for key in keys:
- flags_new[key] = flags_old.get(key, False)
- self.setFlags(flags_new)
- def deleteFlags(self):
- for i in reversed(range(self.flagsLayout.count())):
- item = self.flagsLayout.itemAt(i).widget()
- self.flagsLayout.removeWidget(item)
- item.setParent(None)
- def resetFlags(self, label=""):
- flags = {}
- for pattern, keys in self._flags.items():
- if re.match(pattern, label):
- for key in keys:
- flags[key] = False
- self.setFlags(flags)
- def setFlags(self, flags):
- self.deleteFlags()
- for key in flags:
- item = QtWidgets.QCheckBox(key, self)
- item.setChecked(flags[key])
- self.flagsLayout.addWidget(item)
- item.show()
- def getFlags(self):
- flags = {}
- for i in range(self.flagsLayout.count()):
- item = self.flagsLayout.itemAt(i).widget()
- flags[item.text()] = item.isChecked()
- return flags
- def popUp(self, text=None, move=True, flags=None):
- if self._fit_to_content["row"]:
- self.labelList.setMinimumHeight(
- self.labelList.sizeHintForRow(0) * self.labelList.count() + 2
- )
- if self._fit_to_content["column"]:
- self.labelList.setMinimumWidth(
- self.labelList.sizeHintForColumn(0) + 2
- )
- # if text is None, the previous label in self.edit is kept
- if text is None:
- text = self.edit.text()
- if flags:
- self.setFlags(flags)
- else:
- self.resetFlags(text)
- self.edit.setText(text)
- self.edit.setSelection(0, len(text))
- items = self.labelList.findItems(text, QtCore.Qt.MatchFixedString)
- if items:
- if len(items) != 1:
- self.labelList.setCurrentItem(items[0])
- row = self.labelList.row(items[0])
- self.edit.completer().setCurrentRow(row)
- self.edit.setFocus(QtCore.Qt.PopupFocusReason)
- if move:
- self.move(QtGui.QCursor.pos())
- if self.exec_():
- return self.edit.text(), self.getFlags()
- else:
- return None, None
|