label_dialog.py 8.0 KB
Newer Older
K
Kentaro Wada 已提交
1 2
import re

3
from qtpy import QT_VERSION
4 5 6
from qtpy import QtCore
from qtpy import QtGui
from qtpy import QtWidgets
7

8
from labelme.logger import logger
K
Kentaro Wada 已提交
9
import labelme.utils
10

11

K
Kentaro Wada 已提交
12
QT5 = QT_VERSION[0] == "5"
K
Kentaro Wada 已提交
13 14


15
# TODO(unknown):
16 17
# - Calculate optimal position so as not to go out of screen area.

18 19

class LabelQLineEdit(QtWidgets.QLineEdit):
20 21 22 23
    def setListWidget(self, list_widget):
        self.list_widget = list_widget

    def keyPressEvent(self, e):
24
        if e.key() in [QtCore.Qt.Key_Up, QtCore.Qt.Key_Down]:
25 26 27 28 29
            self.list_widget.keyPressEvent(e)
        else:
            super(LabelQLineEdit, self).keyPressEvent(e)


30
class LabelDialog(QtWidgets.QDialog):
K
Kentaro Wada 已提交
31 32 33 34 35 36 37 38 39 40 41
    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,
    ):
K
Kentaro Wada 已提交
42
        if fit_to_content is None:
K
Kentaro Wada 已提交
43
            fit_to_content = {"row": False, "column": True}
K
Kentaro Wada 已提交
44 45
        self._fit_to_content = fit_to_content

46
        super(LabelDialog, self).__init__(parent)
47
        self.edit = LabelQLineEdit()
K
Kentaro Wada 已提交
48
        self.edit.setPlaceholderText(text)
K
Kentaro Wada 已提交
49
        self.edit.setValidator(labelme.utils.labelValidator())
50
        self.edit.editingFinished.connect(self.postProcess)
51 52
        if flags:
            self.edit.textChanged.connect(self.updateFlags)
K
Kentaro Wada 已提交
53
        self.edit_group_id = QtWidgets.QLineEdit()
K
Kentaro Wada 已提交
54
        self.edit_group_id.setPlaceholderText("Group ID")
K
Kentaro Wada 已提交
55
        self.edit_group_id.setValidator(
K
Kentaro Wada 已提交
56
            QtGui.QRegExpValidator(QtCore.QRegExp(r"\d*"), None)
K
Kentaro Wada 已提交
57
        )
58
        layout = QtWidgets.QVBoxLayout()
59
        if show_text_field:
K
Kentaro Wada 已提交
60 61 62 63
            layout_edit = QtWidgets.QHBoxLayout()
            layout_edit.addWidget(self.edit, 6)
            layout_edit.addWidget(self.edit_group_id, 2)
            layout.addLayout(layout_edit)
K
Kentaro Wada 已提交
64
        # buttons
K
Kentaro Wada 已提交
65 66 67 68 69
        self.buttonBox = bb = QtWidgets.QDialogButtonBox(
            QtWidgets.QDialogButtonBox.Ok | QtWidgets.QDialogButtonBox.Cancel,
            QtCore.Qt.Horizontal,
            self,
        )
K
Kentaro Wada 已提交
70 71
        bb.button(bb.Ok).setIcon(labelme.utils.newIcon("done"))
        bb.button(bb.Cancel).setIcon(labelme.utils.newIcon("undo"))
72 73 74
        bb.accepted.connect(self.validate)
        bb.rejected.connect(self.reject)
        layout.addWidget(bb)
K
Kentaro Wada 已提交
75
        # label_list
76
        self.labelList = QtWidgets.QListWidget()
K
Kentaro Wada 已提交
77
        if self._fit_to_content["row"]:
K
Kentaro Wada 已提交
78 79 80
            self.labelList.setHorizontalScrollBarPolicy(
                QtCore.Qt.ScrollBarAlwaysOff
            )
K
Kentaro Wada 已提交
81
        if self._fit_to_content["column"]:
K
Kentaro Wada 已提交
82 83 84
            self.labelList.setVerticalScrollBarPolicy(
                QtCore.Qt.ScrollBarAlwaysOff
            )
K
Kentaro Wada 已提交
85
        self._sort_labels = sort_labels
K
Kentaro Wada 已提交
86 87
        if labels:
            self.labelList.addItems(labels)
K
Kentaro Wada 已提交
88 89 90
        if self._sort_labels:
            self.labelList.sortItems()
        else:
91
            self.labelList.setDragDropMode(
K
Kentaro Wada 已提交
92 93
                QtWidgets.QAbstractItemView.InternalMove
            )
K
Kentaro Wada 已提交
94
        self.labelList.currentItemChanged.connect(self.labelSelected)
95
        self.labelList.itemDoubleClicked.connect(self.labelDoubleClicked)
96
        self.edit.setListWidget(self.labelList)
K
Kentaro Wada 已提交
97
        layout.addWidget(self.labelList)
98
        # label_flags
99 100 101 102 103 104 105
        if flags is None:
            flags = {}
        self._flags = flags
        self.flagsLayout = QtWidgets.QVBoxLayout()
        self.resetFlags()
        layout.addItem(self.flagsLayout)
        self.edit.textChanged.connect(self.updateFlags)
106
        self.setLayout(layout)
K
Kentaro Wada 已提交
107
        # completion
K
Kentaro Wada 已提交
108
        completer = QtWidgets.QCompleter()
K
Kentaro Wada 已提交
109
        if not QT5 and completion != "startswith":
110 111 112 113
            logger.warn(
                "completion other than 'startswith' is only "
                "supported with Qt5. Using 'startswith'"
            )
K
Kentaro Wada 已提交
114 115
            completion = "startswith"
        if completion == "startswith":
K
Kentaro Wada 已提交
116
            completer.setCompletionMode(QtWidgets.QCompleter.InlineCompletion)
117 118
            # Default settings.
            # completer.setFilterMode(QtCore.Qt.MatchStartsWith)
K
Kentaro Wada 已提交
119
        elif completion == "contains":
K
Kentaro Wada 已提交
120
            completer.setCompletionMode(QtWidgets.QCompleter.PopupCompletion)
121
            completer.setFilterMode(QtCore.Qt.MatchContains)
K
Kentaro Wada 已提交
122
        else:
K
Kentaro Wada 已提交
123
            raise ValueError("Unsupported completion: {}".format(completion))
K
Kentaro Wada 已提交
124 125
        completer.setModel(self.labelList.model())
        self.edit.setCompleter(completer)
126

K
Kentaro Wada 已提交
127
    def addLabelHistory(self, label):
128
        if self.labelList.findItems(label, QtCore.Qt.MatchExactly):
K
Kentaro Wada 已提交
129 130
            return
        self.labelList.addItem(label)
K
Kentaro Wada 已提交
131 132
        if self._sort_labels:
            self.labelList.sortItems()
K
Kentaro Wada 已提交
133 134 135 136

    def labelSelected(self, item):
        self.edit.setText(item.text())

137
    def validate(self):
138
        text = self.edit.text()
K
Kentaro Wada 已提交
139
        if hasattr(text, "strip"):
140
            text = text.strip()
K
Kentaro Wada 已提交
141
        else:
142 143 144
            text = text.trimmed()
        if text:
            self.accept()
145

146 147 148
    def labelDoubleClicked(self, item):
        self.validate()

149
    def postProcess(self):
150
        text = self.edit.text()
K
Kentaro Wada 已提交
151
        if hasattr(text, "strip"):
152
            text = text.strip()
K
Kentaro Wada 已提交
153
        else:
154 155
            text = text.trimmed()
        self.edit.setText(text)
156

157 158 159 160 161
    def updateFlags(self, label_new):
        # keep state of shared flags
        flags_old = self.getFlags()

        flags_new = {}
K
Kentaro Wada 已提交
162 163 164 165
        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)
166 167
        self.setFlags(flags_new)

168
    def deleteFlags(self):
169 170 171
        for i in reversed(range(self.flagsLayout.count())):
            item = self.flagsLayout.itemAt(i).widget()
            self.flagsLayout.removeWidget(item)
172 173
            item.setParent(None)

K
Kentaro Wada 已提交
174
    def resetFlags(self, label=""):
K
Kentaro Wada 已提交
175 176 177 178 179
        flags = {}
        for pattern, keys in self._flags.items():
            if re.match(pattern, label):
                for key in keys:
                    flags[key] = False
180
        self.setFlags(flags)
181

182
    def setFlags(self, flags):
183 184 185 186
        self.deleteFlags()
        for key in flags:
            item = QtWidgets.QCheckBox(key, self)
            item.setChecked(flags[key])
187
            self.flagsLayout.addWidget(item)
188
            item.show()
189 190 191

    def getFlags(self):
        flags = {}
192 193 194
        for i in range(self.flagsLayout.count()):
            item = self.flagsLayout.itemAt(i).widget()
            flags[item.text()] = item.isChecked()
195 196
        return flags

K
Kentaro Wada 已提交
197 198 199 200 201 202 203
    def getGroupId(self):
        group_id = self.edit_group_id.text()
        if group_id:
            return int(group_id)
        return None

    def popUp(self, text=None, move=True, flags=None, group_id=None):
K
Kentaro Wada 已提交
204
        if self._fit_to_content["row"]:
K
Kentaro Wada 已提交
205 206 207
            self.labelList.setMinimumHeight(
                self.labelList.sizeHintForRow(0) * self.labelList.count() + 2
            )
K
Kentaro Wada 已提交
208
        if self._fit_to_content["column"]:
K
Kentaro Wada 已提交
209 210 211
            self.labelList.setMinimumWidth(
                self.labelList.sizeHintForColumn(0) + 2
            )
212
        # if text is None, the previous label in self.edit is kept
K
Kentaro Wada 已提交
213 214
        if text is None:
            text = self.edit.text()
215 216 217 218
        if flags:
            self.setFlags(flags)
        else:
            self.resetFlags(text)
K
Kentaro Wada 已提交
219 220
        self.edit.setText(text)
        self.edit.setSelection(0, len(text))
K
Kentaro Wada 已提交
221 222 223 224
        if group_id is None:
            self.edit_group_id.clear()
        else:
            self.edit_group_id.setText(str(group_id))
K
Kentaro Wada 已提交
225 226
        items = self.labelList.findItems(text, QtCore.Qt.MatchFixedString)
        if items:
K
Kentaro Wada 已提交
227 228
            if len(items) != 1:
                logger.warning("Label list has duplicate '{}'".format(text))
K
Kentaro Wada 已提交
229 230 231
            self.labelList.setCurrentItem(items[0])
            row = self.labelList.row(items[0])
            self.edit.completer().setCurrentRow(row)
232
        self.edit.setFocus(QtCore.Qt.PopupFocusReason)
M
Michael Pitidis 已提交
233
        if move:
234
            self.move(QtGui.QCursor.pos())
C
cmerchant 已提交
235
        if self.exec_():
K
Kentaro Wada 已提交
236
            return self.edit.text(), self.getFlags(), self.getGroupId()
237
        else:
K
Kentaro Wada 已提交
238
            return None, None, None