| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606 | from qtpy import QT_VERSIONfrom qtpy import QtCorefrom qtpy import QtGuifrom qtpy import QtWidgetsQT5 = QT_VERSION[0] == '5'from labelme.lib import distancefrom labelme.shape import Shape# TODO(unknown):# - [maybe] Find optimal epsilon value.CURSOR_DEFAULT = QtCore.Qt.ArrowCursorCURSOR_POINT = QtCore.Qt.PointingHandCursorCURSOR_DRAW = QtCore.Qt.CrossCursorCURSOR_MOVE = QtCore.Qt.ClosedHandCursorCURSOR_GRAB = QtCore.Qt.OpenHandCursorclass Canvas(QtWidgets.QWidget):    zoomRequest = QtCore.Signal(int, QtCore.QPoint)    scrollRequest = QtCore.Signal(int, int)    newShape = QtCore.Signal()    selectionChanged = QtCore.Signal(bool)    shapeMoved = QtCore.Signal()    drawingPolygon = QtCore.Signal(bool)    CREATE, EDIT = 0, 1    epsilon = 11.0    def __init__(self, *args, **kwargs):        super(Canvas, self).__init__(*args, **kwargs)        # Initialise local state.        self.mode = self.EDIT        self.shapes = []        self.shapesBackups = []        self.current = None        self.selectedShape = None  # save the selected shape here        self.selectedShapeCopy = None        self.lineColor = QtGui.QColor(0, 0, 255)        self.line = Shape(line_color=self.lineColor)        self.prevPoint = QtCore.QPointF()        self.offsets = QtCore.QPointF(), QtCore.QPointF()        self.scale = 1.0        self.pixmap = QtGui.QPixmap()        self.visible = {}        self._hideBackround = False        self.hideBackround = False        self.hShape = None        self.hVertex = None        self.movingShape = False        self._painter = QtGui.QPainter()        self._cursor = CURSOR_DEFAULT        # Menus:        self.menus = (QtWidgets.QMenu(), QtWidgets.QMenu())        # Set widget options.        self.setMouseTracking(True)        self.setFocusPolicy(QtCore.Qt.WheelFocus)    def storeShapes(self):        shapesBackup = []        for shape in self.shapes:            shapesBackup.append(shape.copy())        if len(self.shapesBackups) >= 10:            self.shapesBackups = self.shapesBackups[-9:]        self.shapesBackups.append(shapesBackup)    @property    def isShapeRestorable(self):        if len(self.shapesBackups) < 2:            return False        return True    def restoreShape(self):        if not self.isShapeRestorable:            return        self.shapesBackups.pop()  # latest        shapesBackup = self.shapesBackups.pop()        self.shapes = shapesBackup        self.storeShapes()        self.repaint()    def enterEvent(self, ev):        self.overrideCursor(self._cursor)    def leaveEvent(self, ev):        self.restoreCursor()    def focusOutEvent(self, ev):        self.restoreCursor()    def isVisible(self, shape):        return self.visible.get(shape, True)    def drawing(self):        return self.mode == self.CREATE    def editing(self):        return self.mode == self.EDIT    def setEditing(self, value=True):        self.mode = self.EDIT if value else self.CREATE        if not value:  # Create            self.unHighlight()            self.deSelectShape()    def unHighlight(self):        if self.hShape:            self.hShape.highlightClear()        self.hVertex = self.hShape = None    def selectedVertex(self):        return self.hVertex is not None    def mouseMoveEvent(self, ev):        """Update line with last point and current coordinates."""        if QT5:            pos = self.transformPos(ev.pos())        else:            pos = self.transformPos(ev.posF())        self.restoreCursor()        # Polygon drawing.        if self.drawing():            self.overrideCursor(CURSOR_DRAW)            if self.current:                color = self.lineColor                if self.outOfPixmap(pos):                    # Don't allow the user to draw outside the pixmap.                    # Project the point to the pixmap's edges.                    pos = self.intersectionPoint(self.current[-1], pos)                elif len(self.current) > 1 and \                        self.closeEnough(pos, self.current[0]):                    # Attract line to starting point and                    # colorise to alert the user.                    pos = self.current[0]                    color = self.current.line_color                    self.overrideCursor(CURSOR_POINT)                    self.current.highlightVertex(0, Shape.NEAR_VERTEX)                self.line[0] = self.current[-1]                self.line[1] = pos                self.line.line_color = color                self.repaint()                self.current.highlightClear()            return        # Polygon copy moving.        if QtCore.Qt.RightButton & ev.buttons():            if self.selectedShapeCopy and self.prevPoint:                self.overrideCursor(CURSOR_MOVE)                self.boundedMoveShape(self.selectedShapeCopy, pos)                self.repaint()            elif self.selectedShape:                self.selectedShapeCopy = self.selectedShape.copy()                self.repaint()            return        # Polygon/Vertex moving.        self.movingShape = False        if QtCore.Qt.LeftButton & ev.buttons():            if self.selectedVertex():                self.boundedMoveVertex(pos)                self.repaint()                self.movingShape = True            elif self.selectedShape and self.prevPoint:                self.overrideCursor(CURSOR_MOVE)                self.boundedMoveShape(self.selectedShape, pos)                self.repaint()                self.movingShape = True            return        # Just hovering over the canvas, 2 posibilities:        # - Highlight shapes        # - Highlight vertex        # Update shape/vertex fill and tooltip value accordingly.        self.setToolTip("Image")        for shape in reversed([s for s in self.shapes if self.isVisible(s)]):            # Look for a nearby vertex to highlight. If that fails,            # check if we happen to be inside a shape.            index = shape.nearestVertex(pos, self.epsilon)            if index is not None:                if self.selectedVertex():                    self.hShape.highlightClear()                self.hVertex, self.hShape = index, shape                shape.highlightVertex(index, shape.MOVE_VERTEX)                self.overrideCursor(CURSOR_POINT)                self.setToolTip("Click & drag to move point")                self.setStatusTip(self.toolTip())                self.update()                break            elif shape.containsPoint(pos):                if self.selectedVertex():                    self.hShape.highlightClear()                self.hVertex, self.hShape = None, shape                self.setToolTip(                    "Click & drag to move shape '%s'" % shape.label)                self.setStatusTip(self.toolTip())                self.overrideCursor(CURSOR_GRAB)                self.update()                break        else:  # Nothing found, clear highlights, reset state.            if self.hShape:                self.hShape.highlightClear()                self.update()            self.hVertex, self.hShape = None, None    def mousePressEvent(self, ev):        if QT5:            pos = self.transformPos(ev.pos())        else:            pos = self.transformPos(ev.posF())        if ev.button() == QtCore.Qt.LeftButton:            if self.drawing():                if self.current:                    self.current.addPoint(self.line[1])                    self.line[0] = self.current[-1]                    if self.current.isClosed():                        self.finalise()                elif not self.outOfPixmap(pos):                    self.current = Shape()                    self.current.addPoint(pos)                    self.line.points = [pos, pos]                    self.setHiding()                    self.drawingPolygon.emit(True)                    self.update()            else:                self.selectShapePoint(pos)                self.prevPoint = pos                self.repaint()        elif ev.button() == QtCore.Qt.RightButton and self.editing():            self.selectShapePoint(pos)            self.prevPoint = pos            self.repaint()    def mouseReleaseEvent(self, ev):        if ev.button() == QtCore.Qt.RightButton:            menu = self.menus[bool(self.selectedShapeCopy)]            self.restoreCursor()            if not menu.exec_(self.mapToGlobal(ev.pos()))\               and self.selectedShapeCopy:                # Cancel the move by deleting the shadow copy.                self.selectedShapeCopy = None                self.repaint()        elif ev.button() == QtCore.Qt.LeftButton and self.selectedShape:            self.overrideCursor(CURSOR_GRAB)        if self.movingShape:            self.storeShapes()            self.shapeMoved.emit()    def endMove(self, copy=False):        assert self.selectedShape and self.selectedShapeCopy        shape = self.selectedShapeCopy        # del shape.fill_color        # del shape.line_color        if copy:            self.shapes.append(shape)            self.selectedShape.selected = False            self.selectedShape = shape            self.repaint()        else:            shape.label = self.selectedShape.label            self.deleteSelected()            self.shapes.append(shape)        self.storeShapes()        self.selectedShapeCopy = None    def hideBackroundShapes(self, value):        self.hideBackround = value        if self.selectedShape:            # Only hide other shapes if there is a current selection.            # Otherwise the user will not be able to select a shape.            self.setHiding(True)            self.repaint()    def setHiding(self, enable=True):        self._hideBackround = self.hideBackround if enable else False    def canCloseShape(self):        return self.drawing() and self.current and len(self.current) > 2    def mouseDoubleClickEvent(self, ev):        # We need at least 4 points here, since the mousePress handler        # adds an extra one before this handler is called.        if self.canCloseShape() and len(self.current) > 3:            self.current.popPoint()            self.finalise()    def selectShape(self, shape):        self.deSelectShape()        shape.selected = True        self.selectedShape = shape        self.setHiding()        self.selectionChanged.emit(True)        self.update()    def selectShapePoint(self, point):        """Select the first shape created which contains this point."""        self.deSelectShape()        if self.selectedVertex():  # A vertex is marked for selection.            index, shape = self.hVertex, self.hShape            shape.highlightVertex(index, shape.MOVE_VERTEX)            return        for shape in reversed(self.shapes):            if self.isVisible(shape) and shape.containsPoint(point):                shape.selected = True                self.selectedShape = shape                self.calculateOffsets(shape, point)                self.setHiding()                self.selectionChanged.emit(True)                return    def calculateOffsets(self, shape, point):        rect = shape.boundingRect()        x1 = rect.x() - point.x()        y1 = rect.y() - point.y()        x2 = (rect.x() + rect.width()) - point.x()        y2 = (rect.y() + rect.height()) - point.y()        self.offsets = QtCore.QPointF(x1, y1), QtCore.QPointF(x2, y2)    def boundedMoveVertex(self, pos):        index, shape = self.hVertex, self.hShape        point = shape[index]        if self.outOfPixmap(pos):            pos = self.intersectionPoint(point, pos)        shape.moveVertexBy(index, pos - point)    def boundedMoveShape(self, shape, pos):        if self.outOfPixmap(pos):            return False  # No need to move        o1 = pos + self.offsets[0]        if self.outOfPixmap(o1):            pos -= QtCore.QPointF(min(0, o1.x()), min(0, o1.y()))        o2 = pos + self.offsets[1]        if self.outOfPixmap(o2):            pos += QtCore.QPointF(min(0, self.pixmap.width() - o2.x()),                                  min(0, self.pixmap.height() - o2.y()))        # XXX: The next line tracks the new position of the cursor        # relative to the shape, but also results in making it        # a bit "shaky" when nearing the border and allows it to        # go outside of the shape's area for some reason.        # self.calculateOffsets(self.selectedShape, pos)        dp = pos - self.prevPoint        if dp:            shape.moveBy(dp)            self.prevPoint = pos            return True        return False    def deSelectShape(self):        if self.selectedShape:            self.selectedShape.selected = False            self.selectedShape = None            self.setHiding(False)            self.selectionChanged.emit(False)            self.update()    def deleteSelected(self):        if self.selectedShape:            shape = self.selectedShape            self.shapes.remove(self.selectedShape)            self.storeShapes()            self.selectedShape = None            self.update()            return shape    def copySelectedShape(self):        if self.selectedShape:            shape = self.selectedShape.copy()            self.deSelectShape()            self.shapes.append(shape)            self.storeShapes()            shape.selected = True            self.selectedShape = shape            self.boundedShiftShape(shape)            return shape    def boundedShiftShape(self, shape):        # Try to move in one direction, and if it fails in another.        # Give up if both fail.        point = shape[0]        offset = QtCore.QPointF(2.0, 2.0)        self.calculateOffsets(shape, point)        self.prevPoint = point        if not self.boundedMoveShape(shape, point - offset):            self.boundedMoveShape(shape, point + offset)    def paintEvent(self, event):        if not self.pixmap:            return super(Canvas, self).paintEvent(event)        p = self._painter        p.begin(self)        p.setRenderHint(QtGui.QPainter.Antialiasing)        p.setRenderHint(QtGui.QPainter.HighQualityAntialiasing)        p.setRenderHint(QtGui.QPainter.SmoothPixmapTransform)        p.scale(self.scale, self.scale)        p.translate(self.offsetToCenter())        p.drawPixmap(0, 0, self.pixmap)        Shape.scale = self.scale        for shape in self.shapes:            if (shape.selected or not self._hideBackround) and \                    self.isVisible(shape):                shape.fill = shape.selected or shape == self.hShape                shape.paint(p)        if self.current:            self.current.paint(p)            self.line.paint(p)        if self.selectedShapeCopy:            self.selectedShapeCopy.paint(p)        p.end()    def transformPos(self, point):        """Convert from widget-logical coordinates to painter-logical ones."""        return point / self.scale - self.offsetToCenter()    def offsetToCenter(self):        s = self.scale        area = super(Canvas, self).size()        w, h = self.pixmap.width() * s, self.pixmap.height() * s        aw, ah = area.width(), area.height()        x = (aw - w) / (2 * s) if aw > w else 0        y = (ah - h) / (2 * s) if ah > h else 0        if QT5:            return QtCore.QPoint(x, y)        else:            return QtCore.QPointF(x, y)    def outOfPixmap(self, p):        w, h = self.pixmap.width(), self.pixmap.height()        return not (0 <= p.x() <= w and 0 <= p.y() <= h)    def finalise(self):        assert self.current        self.current.close()        self.shapes.append(self.current)        self.storeShapes()        self.current = None        self.setHiding(False)        self.newShape.emit()        self.update()    def closeEnough(self, p1, p2):        # d = distance(p1 - p2)        # m = (p1-p2).manhattanLength()        # print "d %.2f, m %d, %.2f" % (d, m, d - m)        return distance(p1 - p2) < self.epsilon    def intersectionPoint(self, p1, p2):        # Cycle through each image edge in clockwise fashion,        # and find the one intersecting the current line segment.        # http://paulbourke.net/geometry/lineline2d/        size = self.pixmap.size()        points = [(0, 0),                  (size.width(), 0),                  (size.width(), size.height()),                  (0, size.height())]        x1, y1 = p1.x(), p1.y()        x2, y2 = p2.x(), p2.y()        d, i, (x, y) = min(self.intersectingEdges((x1, y1), (x2, y2), points))        x3, y3 = points[i]        x4, y4 = points[(i + 1) % 4]        if (x, y) == (x1, y1):            # Handle cases where previous point is on one of the edges.            if x3 == x4:                return QtCore.QPointF(x3, min(max(0, y2), max(y3, y4)))            else:  # y3 == y4                return QtCore.QPointF(min(max(0, x2), max(x3, x4)), y3)        return QtCore.QPointF(x, y)    def intersectingEdges(self, point1, point2, points):        """Find intersecting edges.        For each edge formed by `points', yield the intersection        with the line segment `(x1,y1) - (x2,y2)`, if it exists.        Also return the distance of `(x2,y2)' to the middle of the        edge along with its index, so that the one closest can be chosen.        """        (x1, y1) = point1        (x2, y2) = point2        for i in range(4):            x3, y3 = points[i]            x4, y4 = points[(i + 1) % 4]            denom = (y4 - y3) * (x2 - x1) - (x4 - x3) * (y2 - y1)            nua = (x4 - x3) * (y1 - y3) - (y4 - y3) * (x1 - x3)            nub = (x2 - x1) * (y1 - y3) - (y2 - y1) * (x1 - x3)            if denom == 0:                # This covers two cases:                #   nua == nub == 0: Coincident                #   otherwise: Parallel                continue            ua, ub = nua / denom, nub / denom            if 0 <= ua <= 1 and 0 <= ub <= 1:                x = x1 + ua * (x2 - x1)                y = y1 + ua * (y2 - y1)                m = QtCore.QPointF((x3 + x4) / 2, (y3 + y4) / 2)                d = distance(m - QtCore.QPointF(x2, y2))                yield d, i, (x, y)    # These two, along with a call to adjustSize are required for the    # scroll area.    def sizeHint(self):        return self.minimumSizeHint()    def minimumSizeHint(self):        if self.pixmap:            return self.scale * self.pixmap.size()        return super(Canvas, self).minimumSizeHint()    def wheelEvent(self, ev):        if QT5:            mods = ev.modifiers()            delta = ev.angleDelta()            if QtCore.Qt.ControlModifier == int(mods):                # with Ctrl/Command key                # zoom                self.zoomRequest.emit(delta.y(), ev.pos())            else:                # scroll                self.scrollRequest.emit(delta.x(), QtCore.Qt.Horizontal)                self.scrollRequest.emit(delta.y(), QtCore.Qt.Vertical)        else:            if ev.orientation() == QtCore.Qt.Vertical:                mods = ev.modifiers()                if QtCore.Qt.ControlModifier == int(mods):                    # with Ctrl/Command key                    self.zoomRequest.emit(ev.delta(), ev.pos())                else:                    self.scrollRequest.emit(                        ev.delta(),                        QtCore.Qt.Horizontal                        if (QtCore.Qt.ShiftModifier == int(mods))                        else QtCore.Qt.Vertical)            else:                self.scrollRequest.emit(ev.delta(), QtCore.Qt.Horizontal)        ev.accept()    def keyPressEvent(self, ev):        key = ev.key()        if key == QtCore.Qt.Key_Escape and self.current:            self.current = None            self.drawingPolygon.emit(False)            self.update()        elif key == QtCore.Qt.Key_Return and self.canCloseShape():            self.finalise()    def setLastLabel(self, text):        assert text        self.shapes[-1].label = text        self.shapesBackups.pop()        self.storeShapes()        return self.shapes[-1]    def undoLastLine(self):        assert self.shapes        self.current = self.shapes.pop()        self.current.setOpen()        self.line.points = [self.current[-1], self.current[0]]        self.drawingPolygon.emit(True)    def undoLastPoint(self):        if not self.current or self.current.isClosed():            return        self.current.popPoint()        if len(self.current) > 0:            self.line[0] = self.current[-1]        else:            self.current = None            self.drawingPolygon.emit(False)        self.repaint()    def loadPixmap(self, pixmap):        self.pixmap = pixmap        self.shapes = []        self.repaint()    def loadShapes(self, shapes):        self.shapes = list(shapes)        self.storeShapes()        self.current = None        self.repaint()    def setShapeVisible(self, shape, value):        self.visible[shape] = value        self.repaint()    def overrideCursor(self, cursor):        self.restoreCursor()        self._cursor = cursor        QtWidgets.QApplication.setOverrideCursor(cursor)    def restoreCursor(self):        QtWidgets.QApplication.restoreOverrideCursor()    def resetState(self):        self.restoreCursor()        self.pixmap = None        self.shapesBackups = []        self.update()
 |