label_dialog.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. from qtpy import QT_VERSION
  2. from qtpy import QtCore
  3. from qtpy import QtGui
  4. from qtpy import QtWidgets
  5. QT5 = QT_VERSION[0] == '5' # NOQA
  6. from labelme import logger
  7. import labelme.utils
  8. # TODO(unknown):
  9. # - Calculate optimal position so as not to go out of screen area.
  10. class LabelQLineEdit(QtWidgets.QLineEdit):
  11. def setListWidget(self, list_widget):
  12. self.list_widget = list_widget
  13. def keyPressEvent(self, e):
  14. if e.key() in [QtCore.Qt.Key_Up, QtCore.Qt.Key_Down]:
  15. self.list_widget.keyPressEvent(e)
  16. else:
  17. super(LabelQLineEdit, self).keyPressEvent(e)
  18. class LabelDialog(QtWidgets.QDialog):
  19. def __init__(self, text="Enter object label", parent=None, labels=None,
  20. sort_labels=True, show_text_field=True,
  21. completion='startswith', fit_to_content=None):
  22. if fit_to_content is None:
  23. fit_to_content = {'row': False, 'column': True}
  24. self._fit_to_content = fit_to_content
  25. super(LabelDialog, self).__init__(parent)
  26. self.edit = LabelQLineEdit()
  27. self.edit.setPlaceholderText(text)
  28. self.edit.setValidator(labelme.utils.labelValidator())
  29. self.edit.editingFinished.connect(self.postProcess)
  30. layout = QtWidgets.QVBoxLayout()
  31. if show_text_field:
  32. layout.addWidget(self.edit)
  33. # buttons
  34. self.buttonBox = bb = QtWidgets.QDialogButtonBox(
  35. QtWidgets.QDialogButtonBox.Ok | QtWidgets.QDialogButtonBox.Cancel,
  36. QtCore.Qt.Horizontal,
  37. self,
  38. )
  39. bb.button(bb.Ok).setIcon(labelme.utils.newIcon('done'))
  40. bb.button(bb.Cancel).setIcon(labelme.utils.newIcon('undo'))
  41. bb.accepted.connect(self.validate)
  42. bb.rejected.connect(self.reject)
  43. layout.addWidget(bb)
  44. # label_list
  45. self.labelList = QtWidgets.QListWidget()
  46. if self._fit_to_content['row']:
  47. self.labelList.setHorizontalScrollBarPolicy(
  48. QtCore.Qt.ScrollBarAlwaysOff
  49. )
  50. if self._fit_to_content['column']:
  51. self.labelList.setVerticalScrollBarPolicy(
  52. QtCore.Qt.ScrollBarAlwaysOff
  53. )
  54. self._sort_labels = sort_labels
  55. if labels:
  56. self.labelList.addItems(labels)
  57. if self._sort_labels:
  58. self.labelList.sortItems()
  59. else:
  60. self.labelList.setDragDropMode(
  61. QtWidgets.QAbstractItemView.InternalMove)
  62. self.labelList.currentItemChanged.connect(self.labelSelected)
  63. self.edit.setListWidget(self.labelList)
  64. layout.addWidget(self.labelList)
  65. self.setLayout(layout)
  66. # completion
  67. completer = QtWidgets.QCompleter()
  68. if not QT5 and completion != 'startswith':
  69. logger.warn(
  70. "completion other than 'startswith' is only "
  71. "supported with Qt5. Using 'startswith'"
  72. )
  73. completion = 'startswith'
  74. if completion == 'startswith':
  75. completer.setCompletionMode(QtWidgets.QCompleter.InlineCompletion)
  76. # Default settings.
  77. # completer.setFilterMode(QtCore.Qt.MatchStartsWith)
  78. elif completion == 'contains':
  79. completer.setCompletionMode(QtWidgets.QCompleter.PopupCompletion)
  80. completer.setFilterMode(QtCore.Qt.MatchContains)
  81. else:
  82. raise ValueError('Unsupported completion: {}'.format(completion))
  83. completer.setModel(self.labelList.model())
  84. self.edit.setCompleter(completer)
  85. def addLabelHistory(self, label):
  86. if self.labelList.findItems(label, QtCore.Qt.MatchExactly):
  87. return
  88. self.labelList.addItem(label)
  89. if self._sort_labels:
  90. self.labelList.sortItems()
  91. def labelSelected(self, item):
  92. self.edit.setText(item.text())
  93. def validate(self):
  94. text = self.edit.text()
  95. if hasattr(text, 'strip'):
  96. text = text.strip()
  97. else:
  98. text = text.trimmed()
  99. if text:
  100. self.accept()
  101. def postProcess(self):
  102. text = self.edit.text()
  103. if hasattr(text, 'strip'):
  104. text = text.strip()
  105. else:
  106. text = text.trimmed()
  107. self.edit.setText(text)
  108. def popUp(self, text=None, move=True):
  109. if self._fit_to_content['row']:
  110. self.labelList.setMinimumHeight(
  111. self.labelList.sizeHintForRow(0) * self.labelList.count() + 2
  112. )
  113. if self._fit_to_content['column']:
  114. self.labelList.setMinimumWidth(
  115. self.labelList.sizeHintForColumn(0) + 2
  116. )
  117. # if text is None, the previous label in self.edit is kept
  118. if text is None:
  119. text = self.edit.text()
  120. self.edit.setText(text)
  121. self.edit.setSelection(0, len(text))
  122. items = self.labelList.findItems(text, QtCore.Qt.MatchFixedString)
  123. if items:
  124. if len(items) != 1:
  125. logger.warning("Label list has duplicate '{}'".format(text))
  126. self.labelList.setCurrentItem(items[0])
  127. row = self.labelList.row(items[0])
  128. self.edit.completer().setCurrentRow(row)
  129. self.edit.setFocus(QtCore.Qt.PopupFocusReason)
  130. if move:
  131. self.move(QtGui.QCursor.pos())
  132. return self.edit.text() if self.exec_() else None