label_dialog.py 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  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.logger 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, flags=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. if flags:
  31. self.edit.textChanged.connect(self.updateFlags)
  32. layout = QtWidgets.QVBoxLayout()
  33. if show_text_field:
  34. layout.addWidget(self.edit)
  35. # buttons
  36. self.buttonBox = bb = QtWidgets.QDialogButtonBox(
  37. QtWidgets.QDialogButtonBox.Ok | QtWidgets.QDialogButtonBox.Cancel,
  38. QtCore.Qt.Horizontal,
  39. self,
  40. )
  41. bb.button(bb.Ok).setIcon(labelme.utils.newIcon('done'))
  42. bb.button(bb.Cancel).setIcon(labelme.utils.newIcon('undo'))
  43. bb.accepted.connect(self.validate)
  44. bb.rejected.connect(self.reject)
  45. layout.addWidget(bb)
  46. # label_list
  47. self.labelList = QtWidgets.QListWidget()
  48. if self._fit_to_content['row']:
  49. self.labelList.setHorizontalScrollBarPolicy(
  50. QtCore.Qt.ScrollBarAlwaysOff
  51. )
  52. if self._fit_to_content['column']:
  53. self.labelList.setVerticalScrollBarPolicy(
  54. QtCore.Qt.ScrollBarAlwaysOff
  55. )
  56. self._sort_labels = sort_labels
  57. if labels:
  58. self.labelList.addItems(labels)
  59. if self._sort_labels:
  60. self.labelList.sortItems()
  61. else:
  62. self.labelList.setDragDropMode(
  63. QtWidgets.QAbstractItemView.InternalMove)
  64. self.labelList.currentItemChanged.connect(self.labelSelected)
  65. self.edit.setListWidget(self.labelList)
  66. layout.addWidget(self.labelList)
  67. # label_flags
  68. self.flags = flags
  69. self.label_flags = None
  70. if flags:
  71. self.label_flags = QtWidgets.QVBoxLayout()
  72. self.resetFlags()
  73. layout.addItem(self.label_flags)
  74. self.setLayout(layout)
  75. # completion
  76. completer = QtWidgets.QCompleter()
  77. if not QT5 and completion != 'startswith':
  78. logger.warn(
  79. "completion other than 'startswith' is only "
  80. "supported with Qt5. Using 'startswith'"
  81. )
  82. completion = 'startswith'
  83. if completion == 'startswith':
  84. completer.setCompletionMode(QtWidgets.QCompleter.InlineCompletion)
  85. # Default settings.
  86. # completer.setFilterMode(QtCore.Qt.MatchStartsWith)
  87. elif completion == 'contains':
  88. completer.setCompletionMode(QtWidgets.QCompleter.PopupCompletion)
  89. completer.setFilterMode(QtCore.Qt.MatchContains)
  90. else:
  91. raise ValueError('Unsupported completion: {}'.format(completion))
  92. completer.setModel(self.labelList.model())
  93. self.edit.setCompleter(completer)
  94. def addLabelHistory(self, label):
  95. if self.labelList.findItems(label, QtCore.Qt.MatchExactly):
  96. return
  97. self.labelList.addItem(label)
  98. if self._sort_labels:
  99. self.labelList.sortItems()
  100. def labelSelected(self, item):
  101. self.edit.setText(item.text())
  102. def updateFlags(self, text):
  103. flags = self.getFlags()
  104. newFlags = {}
  105. for label in ["__all__", text]:
  106. if label in self.flags:
  107. for key in self.flags[label]:
  108. newFlags[key] = False if key not in flags else flags[key]
  109. self.setFlags(newFlags)
  110. def validate(self):
  111. text = self.edit.text()
  112. if hasattr(text, 'strip'):
  113. text = text.strip()
  114. else:
  115. text = text.trimmed()
  116. if text:
  117. self.accept()
  118. def postProcess(self):
  119. text = self.edit.text()
  120. if hasattr(text, 'strip'):
  121. text = text.strip()
  122. else:
  123. text = text.trimmed()
  124. self.edit.setText(text)
  125. def deleteFlags(self):
  126. for i in reversed(range(self.label_flags.count())):
  127. item = self.label_flags.itemAt(i).widget()
  128. self.label_flags.removeWidget(item)
  129. item.setParent(None)
  130. def resetFlags(self, text=''):
  131. self.deleteFlags()
  132. # Add all flags
  133. for label in ["__all__", text]:
  134. if label in self.flags:
  135. for key in self.flags[label]:
  136. item = QtWidgets.QCheckBox(key, self)
  137. self.label_flags.addWidget(item)
  138. item.show()
  139. def setFlags(self, flags, text=''):
  140. self.deleteFlags()
  141. # Add flags not set
  142. for label in ["__all__", text]:
  143. if label in self.flags:
  144. for key in self.flags[label]:
  145. if key not in flags:
  146. item = QtWidgets.QCheckBox(key, self)
  147. self.label_flags.addWidget(item)
  148. item.show()
  149. # Add set flags
  150. for key in flags:
  151. item = QtWidgets.QCheckBox(key, self)
  152. item.setChecked(flags[key])
  153. self.label_flags.addWidget(item)
  154. item.show()
  155. def getFlags(self):
  156. flags = {}
  157. for i in range(self.label_flags.count()):
  158. item = self.label_flags.itemAt(i).widget()
  159. flags[item.text()] = True if item.isChecked() else False
  160. return flags
  161. def popUp(self, text=None, move=True, flags=None):
  162. if self._fit_to_content['row']:
  163. self.labelList.setMinimumHeight(
  164. self.labelList.sizeHintForRow(0) * self.labelList.count() + 2
  165. )
  166. if self._fit_to_content['column']:
  167. self.labelList.setMinimumWidth(
  168. self.labelList.sizeHintForColumn(0) + 2
  169. )
  170. # if text is None, the previous label in self.edit is kept
  171. if text is None:
  172. text = self.edit.text()
  173. if self.label_flags:
  174. if flags:
  175. self.setFlags(flags)
  176. else:
  177. self.resetFlags(text)
  178. self.edit.setText(text)
  179. self.edit.setSelection(0, len(text))
  180. items = self.labelList.findItems(text, QtCore.Qt.MatchFixedString)
  181. if items:
  182. if len(items) != 1:
  183. logger.warning("Label list has duplicate '{}'".format(text))
  184. self.labelList.setCurrentItem(items[0])
  185. row = self.labelList.row(items[0])
  186. self.edit.completer().setCurrentRow(row)
  187. self.edit.setFocus(QtCore.Qt.PopupFocusReason)
  188. if move:
  189. self.move(QtGui.QCursor.pos())
  190. if self._exec():
  191. return self.edit.text(), self.getFlags() if self.flags else None
  192. else:
  193. return None, None