2020-12-11 19:23:27 +08:00
|
|
|
|
# Copyright (c) <2015-Present> Tzutalin
|
|
|
|
|
# Copyright (C) 2013 MIT, Computer Science and Artificial Intelligence Laboratory. Bryan Russell, Antonio Torralba,
|
|
|
|
|
# William T. Freeman. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and
|
|
|
|
|
# associated documentation files (the "Software"), to deal in the Software without restriction, including without
|
|
|
|
|
# limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the
|
|
|
|
|
# Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
|
|
|
|
# The above copyright notice and this permission notice shall be included in all copies or substantial portions of
|
|
|
|
|
# the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT
|
|
|
|
|
# NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT
|
|
|
|
|
# SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
|
|
|
|
|
# CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
|
# THE SOFTWARE.
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
from PyQt5.QtGui import *
|
|
|
|
|
from PyQt5.QtCore import *
|
|
|
|
|
from PyQt5.QtWidgets import *
|
|
|
|
|
except ImportError:
|
|
|
|
|
from PyQt4.QtGui import *
|
|
|
|
|
from PyQt4.QtCore import *
|
|
|
|
|
|
|
|
|
|
#from PyQt4.QtOpenGL import *
|
|
|
|
|
|
|
|
|
|
from libs.shape import Shape
|
|
|
|
|
from libs.utils import distance
|
2021-07-28 21:12:00 +08:00
|
|
|
|
import copy
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
CURSOR_DEFAULT = Qt.ArrowCursor
|
|
|
|
|
CURSOR_POINT = Qt.PointingHandCursor
|
|
|
|
|
CURSOR_DRAW = Qt.CrossCursor
|
|
|
|
|
CURSOR_MOVE = Qt.ClosedHandCursor
|
|
|
|
|
CURSOR_GRAB = Qt.OpenHandCursor
|
|
|
|
|
|
|
|
|
|
# class Canvas(QGLWidget):
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Canvas(QWidget):
|
|
|
|
|
zoomRequest = pyqtSignal(int)
|
|
|
|
|
scrollRequest = pyqtSignal(int, int)
|
|
|
|
|
newShape = pyqtSignal()
|
2021-02-05 12:05:11 +08:00
|
|
|
|
# selectionChanged = pyqtSignal(bool)
|
|
|
|
|
selectionChanged = pyqtSignal(list)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
shapeMoved = pyqtSignal()
|
|
|
|
|
drawingPolygon = pyqtSignal(bool)
|
|
|
|
|
|
|
|
|
|
CREATE, EDIT = list(range(2))
|
|
|
|
|
_fill_drawing = False # draw shadows
|
|
|
|
|
|
2021-04-14 21:24:37 +08:00
|
|
|
|
epsilon = 5.0
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
super(Canvas, self).__init__(*args, **kwargs)
|
|
|
|
|
# Initialise local state.
|
|
|
|
|
self.mode = self.EDIT
|
|
|
|
|
self.shapes = []
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.shapesBackups = []
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.current = None
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.selectedShapes = []
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.selectedShape = None # save the selected shape here
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.selectedShapesCopy = []
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.drawingLineColor = QColor(0, 0, 255)
|
|
|
|
|
self.drawingRectColor = QColor(0, 0, 255)
|
|
|
|
|
self.line = Shape(line_color=self.drawingLineColor)
|
|
|
|
|
self.prevPoint = QPointF()
|
|
|
|
|
self.offsets = QPointF(), QPointF()
|
|
|
|
|
self.scale = 1.0
|
|
|
|
|
self.pixmap = QPixmap()
|
|
|
|
|
self.visible = {}
|
|
|
|
|
self._hideBackround = False
|
|
|
|
|
self.hideBackround = False
|
|
|
|
|
self.hShape = None
|
|
|
|
|
self.hVertex = None
|
|
|
|
|
self._painter = QPainter()
|
|
|
|
|
self._cursor = CURSOR_DEFAULT
|
|
|
|
|
# Menus:
|
|
|
|
|
self.menus = (QMenu(), QMenu())
|
|
|
|
|
# Set widget options.
|
|
|
|
|
self.setMouseTracking(True)
|
|
|
|
|
self.setFocusPolicy(Qt.WheelFocus)
|
|
|
|
|
self.verified = False
|
|
|
|
|
self.drawSquare = False
|
|
|
|
|
self.fourpoint = True # ADD
|
|
|
|
|
self.pointnum = 0
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.movingShape = False
|
2021-07-28 21:12:00 +08:00
|
|
|
|
self.selectCountShape = False
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
#initialisation for panning
|
|
|
|
|
self.pan_initial_pos = QPoint()
|
|
|
|
|
|
|
|
|
|
def setDrawingColor(self, qColor):
|
|
|
|
|
self.drawingLineColor = qColor
|
|
|
|
|
self.drawingRectColor = qColor
|
|
|
|
|
|
|
|
|
|
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()
|
|
|
|
|
self.prevPoint = QPointF()
|
|
|
|
|
self.repaint()
|
|
|
|
|
|
|
|
|
|
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."""
|
|
|
|
|
pos = self.transformPos(ev.pos())
|
|
|
|
|
|
|
|
|
|
# Update coordinates in status bar if image is opened
|
|
|
|
|
window = self.parent().window()
|
|
|
|
|
if window.filePath is not None:
|
|
|
|
|
self.parent().window().labelCoordinates.setText(
|
|
|
|
|
'X: %d; Y: %d' % (pos.x(), pos.y()))
|
|
|
|
|
|
|
|
|
|
# Polygon drawing.
|
|
|
|
|
if self.drawing():
|
|
|
|
|
self.overrideCursor(CURSOR_DRAW) # ?
|
|
|
|
|
if self.current:
|
|
|
|
|
# Display annotation width and height while drawing
|
|
|
|
|
currentWidth = abs(self.current[0].x() - pos.x())
|
|
|
|
|
currentHeight = abs(self.current[0].y() - pos.y())
|
|
|
|
|
self.parent().window().labelCoordinates.setText(
|
|
|
|
|
'Width: %d, Height: %d / X: %d; Y: %d' % (currentWidth, currentHeight, pos.x(), pos.y()))
|
|
|
|
|
|
|
|
|
|
color = self.drawingLineColor
|
|
|
|
|
if self.outOfPixmap(pos):
|
|
|
|
|
# Don't allow the user to draw outside the pixmap.
|
|
|
|
|
# Clip the coordinates to 0 or max,
|
|
|
|
|
# if they are outside the range [0, max]
|
|
|
|
|
size = self.pixmap.size()
|
|
|
|
|
clipped_x = min(max(0, pos.x()), size.width())
|
|
|
|
|
clipped_y = min(max(0, pos.y()), size.height())
|
|
|
|
|
pos = QPointF(clipped_x, clipped_y)
|
2021-02-05 12:05:11 +08:00
|
|
|
|
|
2021-02-07 17:28:10 +08:00
|
|
|
|
elif len(self.current) > 1 and self.closeEnough(pos, self.current[0]):
|
2020-12-11 19:23:27 +08:00
|
|
|
|
# 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)
|
2021-02-07 17:28:10 +08:00
|
|
|
|
|
|
|
|
|
if self.drawSquare:
|
|
|
|
|
self.line.points = [self.current[0], pos]
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.line.close()
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
elif self.fourpoint:
|
|
|
|
|
self.line[0] = self.current[-1]
|
|
|
|
|
self.line[1] = pos
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
self.line[1] = pos # pos is the mouse's current position
|
|
|
|
|
|
|
|
|
|
self.line.line_color = color
|
|
|
|
|
self.prevPoint = QPointF() # ?
|
|
|
|
|
self.current.highlightClear()
|
|
|
|
|
else:
|
|
|
|
|
self.prevPoint = pos
|
|
|
|
|
self.repaint()
|
|
|
|
|
return
|
2021-02-07 17:28:10 +08:00
|
|
|
|
|
2020-12-11 19:23:27 +08:00
|
|
|
|
# Polygon copy moving.
|
|
|
|
|
if Qt.RightButton & ev.buttons():
|
2021-02-05 12:05:11 +08:00
|
|
|
|
if self.selectedShapesCopy and self.prevPoint:
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.overrideCursor(CURSOR_MOVE)
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.boundedMoveShape(self.selectedShapesCopy, pos)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.repaint()
|
2021-02-05 12:05:11 +08:00
|
|
|
|
elif self.selectedShapes:
|
|
|
|
|
self.selectedShapesCopy = [
|
|
|
|
|
s.copy() for s in self.selectedShapes
|
|
|
|
|
]
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.repaint()
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
# Polygon/Vertex moving.
|
|
|
|
|
if Qt.LeftButton & ev.buttons():
|
|
|
|
|
if self.selectedVertex():
|
|
|
|
|
self.boundedMoveVertex(pos)
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.shapeMoved.emit()
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.repaint()
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.movingShape = True
|
|
|
|
|
elif self.selectedShapes and self.prevPoint:
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.overrideCursor(CURSOR_MOVE)
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.boundedMoveShape(self.selectedShapes, pos)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.shapeMoved.emit()
|
|
|
|
|
self.repaint()
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.movingShape = True
|
2021-02-07 17:28:10 +08:00
|
|
|
|
else:
|
2020-12-11 19:23:27 +08:00
|
|
|
|
#pan
|
|
|
|
|
delta_x = pos.x() - self.pan_initial_pos.x()
|
|
|
|
|
delta_y = pos.y() - self.pan_initial_pos.y()
|
|
|
|
|
self.scrollRequest.emit(delta_x, Qt.Horizontal)
|
|
|
|
|
self.scrollRequest.emit(delta_y, Qt.Vertical)
|
|
|
|
|
self.update()
|
|
|
|
|
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()
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.hVertex, self.hShape = index, shape
|
2020-12-11 19:23:27 +08:00
|
|
|
|
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
|
|
|
|
|
self.overrideCursor(CURSOR_DEFAULT)
|
|
|
|
|
|
|
|
|
|
def mousePressEvent(self, ev):
|
|
|
|
|
pos = self.transformPos(ev.pos())
|
|
|
|
|
if ev.button() == Qt.LeftButton:
|
|
|
|
|
if self.drawing():
|
|
|
|
|
# self.handleDrawing(pos) # OLD
|
2021-02-05 12:05:11 +08:00
|
|
|
|
if self.current:
|
|
|
|
|
if self.fourpoint: # ADD IF
|
|
|
|
|
# Add point to existing shape.
|
|
|
|
|
# print('Adding points in mousePressEvent is ', self.line[1])
|
|
|
|
|
self.current.addPoint(self.line[1])
|
|
|
|
|
self.line[0] = self.current[-1]
|
|
|
|
|
if self.current.isClosed():
|
|
|
|
|
# print('1111')
|
|
|
|
|
self.finalise()
|
|
|
|
|
elif self.drawSquare: # 增加
|
|
|
|
|
assert len(self.current.points) == 1
|
|
|
|
|
self.current.points = self.line.points
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.finalise()
|
|
|
|
|
elif not self.outOfPixmap(pos):
|
|
|
|
|
# Create new shape.
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.current = Shape()
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.current.addPoint(pos)
|
|
|
|
|
self.line.points = [pos, pos]
|
|
|
|
|
self.setHiding()
|
|
|
|
|
self.drawingPolygon.emit(True)
|
|
|
|
|
self.update()
|
|
|
|
|
|
2021-02-07 17:28:10 +08:00
|
|
|
|
else:
|
2021-02-05 12:05:11 +08:00
|
|
|
|
group_mode = int(ev.modifiers()) == Qt.ControlModifier
|
|
|
|
|
self.selectShapePoint(pos, multiple_selection_mode=group_mode)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.prevPoint = pos
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.pan_initial_pos = pos
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
elif ev.button() == Qt.RightButton and self.editing():
|
2021-02-05 12:05:11 +08:00
|
|
|
|
group_mode = int(ev.modifiers()) == Qt.ControlModifier
|
|
|
|
|
self.selectShapePoint(pos, multiple_selection_mode=group_mode)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.prevPoint = pos
|
|
|
|
|
self.update()
|
|
|
|
|
|
|
|
|
|
def mouseReleaseEvent(self, ev):
|
|
|
|
|
if ev.button() == Qt.RightButton:
|
2021-02-05 12:05:11 +08:00
|
|
|
|
menu = self.menus[bool(self.selectedShapesCopy)]
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.restoreCursor()
|
|
|
|
|
if not menu.exec_(self.mapToGlobal(ev.pos()))\
|
2021-02-05 12:05:11 +08:00
|
|
|
|
and self.selectedShapesCopy:
|
2020-12-11 19:23:27 +08:00
|
|
|
|
# Cancel the move by deleting the shadow copy.
|
2021-02-05 12:05:11 +08:00
|
|
|
|
# self.selectedShapeCopy = None
|
|
|
|
|
self.selectedShapesCopy = []
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.repaint()
|
2021-02-07 17:28:10 +08:00
|
|
|
|
|
2021-02-05 12:05:11 +08:00
|
|
|
|
elif ev.button() == Qt.LeftButton and self.selectedShapes:
|
2020-12-11 19:23:27 +08:00
|
|
|
|
if self.selectedVertex():
|
|
|
|
|
self.overrideCursor(CURSOR_POINT)
|
|
|
|
|
else:
|
|
|
|
|
self.overrideCursor(CURSOR_GRAB)
|
|
|
|
|
|
2021-02-07 17:28:10 +08:00
|
|
|
|
elif ev.button() == Qt.LeftButton and not self.fourpoint:
|
2020-12-11 19:23:27 +08:00
|
|
|
|
pos = self.transformPos(ev.pos())
|
|
|
|
|
if self.drawing():
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.handleDrawing(pos)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
else:
|
|
|
|
|
#pan
|
|
|
|
|
QApplication.restoreOverrideCursor() # ?
|
|
|
|
|
|
2021-02-07 17:28:10 +08:00
|
|
|
|
if self.movingShape and self.hShape:
|
2021-02-05 12:05:11 +08:00
|
|
|
|
index = self.shapes.index(self.hShape)
|
|
|
|
|
if (
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.shapesBackups[-1][index].points
|
2021-02-05 12:05:11 +08:00
|
|
|
|
!= self.shapes[index].points
|
|
|
|
|
):
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.storeShapes()
|
|
|
|
|
self.shapeMoved.emit() # connect to updateBoxlist in PPOCRLabel.py
|
2021-02-05 12:05:11 +08:00
|
|
|
|
|
|
|
|
|
self.movingShape = False
|
|
|
|
|
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
def endMove(self, copy=False):
|
2021-02-05 12:05:11 +08:00
|
|
|
|
assert self.selectedShapes and self.selectedShapesCopy
|
|
|
|
|
assert len(self.selectedShapesCopy) == len(self.selectedShapes)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
if copy:
|
2021-02-05 12:05:11 +08:00
|
|
|
|
for i, shape in enumerate(self.selectedShapesCopy):
|
|
|
|
|
self.shapes.append(shape)
|
|
|
|
|
self.selectedShapes[i].selected = False
|
|
|
|
|
self.selectedShapes[i] = shape
|
2020-12-11 19:23:27 +08:00
|
|
|
|
else:
|
2021-02-05 12:05:11 +08:00
|
|
|
|
for i, shape in enumerate(self.selectedShapesCopy):
|
|
|
|
|
self.selectedShapes[i].points = shape.points
|
|
|
|
|
self.selectedShapesCopy = []
|
|
|
|
|
self.repaint()
|
|
|
|
|
self.storeShapes()
|
|
|
|
|
return True
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
def hideBackroundShapes(self, value):
|
|
|
|
|
self.hideBackround = value
|
2021-02-05 12:05:11 +08:00
|
|
|
|
if self.selectedShapes:
|
2020-12-11 19:23:27 +08:00
|
|
|
|
# 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()
|
|
|
|
|
|
2021-02-07 17:28:10 +08:00
|
|
|
|
def handleDrawing(self, pos):
|
2020-12-11 19:23:27 +08:00
|
|
|
|
if self.current and self.current.reachMaxPoints() is False:
|
|
|
|
|
if self.fourpoint:
|
|
|
|
|
targetPos = self.line[self.pointnum]
|
|
|
|
|
self.current.addPoint(targetPos)
|
|
|
|
|
print('current points in handleDrawing is ', self.line[self.pointnum])
|
|
|
|
|
self.update()
|
|
|
|
|
if self.pointnum == 3:
|
|
|
|
|
self.finalise()
|
|
|
|
|
|
2021-02-07 17:28:10 +08:00
|
|
|
|
else:
|
2020-12-11 19:23:27 +08:00
|
|
|
|
initPos = self.current[0]
|
|
|
|
|
print('initPos', self.current[0])
|
|
|
|
|
minX = initPos.x()
|
|
|
|
|
minY = initPos.y()
|
|
|
|
|
targetPos = self.line[1]
|
|
|
|
|
maxX = targetPos.x()
|
|
|
|
|
maxY = targetPos.y()
|
|
|
|
|
self.current.addPoint(QPointF(maxX, minY))
|
|
|
|
|
self.current.addPoint(targetPos)
|
|
|
|
|
self.current.addPoint(QPointF(minX, maxY))
|
|
|
|
|
self.finalise()
|
|
|
|
|
|
|
|
|
|
elif not self.outOfPixmap(pos):
|
|
|
|
|
print('release')
|
|
|
|
|
self.current = Shape()
|
|
|
|
|
self.current.addPoint(pos)
|
|
|
|
|
self.line.points = [pos, pos]
|
|
|
|
|
self.setHiding()
|
|
|
|
|
self.drawingPolygon.emit(True)
|
|
|
|
|
self.update()
|
|
|
|
|
|
|
|
|
|
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:
|
|
|
|
|
if not self.fourpoint:
|
|
|
|
|
self.current.popPoint()
|
|
|
|
|
self.finalise()
|
|
|
|
|
|
2021-02-05 12:05:11 +08:00
|
|
|
|
def selectShapes(self, shapes):
|
|
|
|
|
for s in shapes: s.seleted = True
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.setHiding()
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.selectionChanged.emit(shapes)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.update()
|
|
|
|
|
|
2021-02-05 12:05:11 +08:00
|
|
|
|
|
|
|
|
|
def selectShapePoint(self, point, multiple_selection_mode):
|
2020-12-11 19:23:27 +08:00
|
|
|
|
"""Select the first shape created which contains this point."""
|
|
|
|
|
if self.selectedVertex(): # A vertex is marked for selection.
|
|
|
|
|
index, shape = self.hVertex, self.hShape
|
2021-02-07 17:28:10 +08:00
|
|
|
|
shape.highlightVertex(index, shape.MOVE_VERTEX)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
return self.hVertex
|
2021-02-05 12:05:11 +08:00
|
|
|
|
else:
|
|
|
|
|
for shape in reversed(self.shapes):
|
|
|
|
|
if self.isVisible(shape) and shape.containsPoint(point):
|
|
|
|
|
self.calculateOffsets(shape, point)
|
|
|
|
|
self.setHiding()
|
|
|
|
|
if multiple_selection_mode:
|
2021-02-07 17:28:10 +08:00
|
|
|
|
if shape not in self.selectedShapes: # list
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.selectionChanged.emit(
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.selectedShapes + [shape]
|
2021-02-05 12:05:11 +08:00
|
|
|
|
)
|
|
|
|
|
else:
|
|
|
|
|
self.selectionChanged.emit([shape])
|
|
|
|
|
return
|
|
|
|
|
self.deSelectShape()
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
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 = QPointF(x1, y1), QPointF(x2, y2)
|
|
|
|
|
|
|
|
|
|
def snapPointToCanvas(self, x, y):
|
|
|
|
|
"""
|
|
|
|
|
Moves a point x,y to within the boundaries of the canvas.
|
|
|
|
|
:return: (x,y,snapped) where snapped is True if x or y were changed, False if not.
|
|
|
|
|
"""
|
|
|
|
|
if x < 0 or x > self.pixmap.width() or y < 0 or y > self.pixmap.height():
|
|
|
|
|
x = max(x, 0)
|
|
|
|
|
y = max(y, 0)
|
|
|
|
|
x = min(x, self.pixmap.width())
|
|
|
|
|
y = min(y, self.pixmap.height())
|
|
|
|
|
return x, y, True
|
|
|
|
|
|
|
|
|
|
return x, y, False
|
|
|
|
|
|
|
|
|
|
def boundedMoveVertex(self, pos):
|
|
|
|
|
index, shape = self.hVertex, self.hShape
|
|
|
|
|
point = shape[index]
|
|
|
|
|
if self.outOfPixmap(pos):
|
|
|
|
|
size = self.pixmap.size()
|
|
|
|
|
clipped_x = min(max(0, pos.x()), size.width())
|
|
|
|
|
clipped_y = min(max(0, pos.y()), size.height())
|
|
|
|
|
pos = QPointF(clipped_x, clipped_y)
|
|
|
|
|
|
|
|
|
|
if self.drawSquare:
|
|
|
|
|
opposite_point_index = (index + 2) % 4
|
|
|
|
|
opposite_point = shape[opposite_point_index]
|
|
|
|
|
|
|
|
|
|
min_size = min(abs(pos.x() - opposite_point.x()), abs(pos.y() - opposite_point.y()))
|
|
|
|
|
directionX = -1 if pos.x() - opposite_point.x() < 0 else 1
|
|
|
|
|
directionY = -1 if pos.y() - opposite_point.y() < 0 else 1
|
|
|
|
|
shiftPos = QPointF(opposite_point.x() + directionX * min_size - point.x(),
|
|
|
|
|
opposite_point.y() + directionY * min_size - point.y())
|
|
|
|
|
else:
|
|
|
|
|
shiftPos = pos - point
|
|
|
|
|
|
2021-02-07 17:28:10 +08:00
|
|
|
|
if [shape[0].x(), shape[0].y(), shape[2].x(), shape[2].y()] \
|
|
|
|
|
== [shape[3].x(),shape[1].y(),shape[1].x(),shape[3].y()]:
|
2021-02-05 12:05:11 +08:00
|
|
|
|
shape.moveVertexBy(index, shiftPos)
|
|
|
|
|
lindex = (index + 1) % 4
|
|
|
|
|
rindex = (index + 3) % 4
|
|
|
|
|
lshift = None
|
|
|
|
|
rshift = None
|
|
|
|
|
if index % 2 == 0:
|
|
|
|
|
rshift = QPointF(shiftPos.x(), 0)
|
|
|
|
|
lshift = QPointF(0, shiftPos.y())
|
|
|
|
|
else:
|
|
|
|
|
lshift = QPointF(shiftPos.x(), 0)
|
|
|
|
|
rshift = QPointF(0, shiftPos.y())
|
|
|
|
|
shape.moveVertexBy(rindex, rshift)
|
|
|
|
|
shape.moveVertexBy(lindex, lshift)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
else:
|
2021-02-05 12:05:11 +08:00
|
|
|
|
shape.moveVertexBy(index, shiftPos)
|
|
|
|
|
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
2021-02-05 12:05:11 +08:00
|
|
|
|
def boundedMoveShape(self, shapes, pos):
|
2021-02-07 17:28:10 +08:00
|
|
|
|
if type(shapes).__name__ != 'list': shapes = [shapes]
|
2020-12-11 19:23:27 +08:00
|
|
|
|
if self.outOfPixmap(pos):
|
|
|
|
|
return False # No need to move
|
|
|
|
|
o1 = pos + self.offsets[0]
|
|
|
|
|
if self.outOfPixmap(o1):
|
|
|
|
|
pos -= QPointF(min(0, o1.x()), min(0, o1.y()))
|
|
|
|
|
o2 = pos + self.offsets[1]
|
|
|
|
|
if self.outOfPixmap(o2):
|
|
|
|
|
pos += QPointF(min(0, self.pixmap.width() - o2.x()),
|
|
|
|
|
min(0, self.pixmap.height() - o2.y()))
|
|
|
|
|
# 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. XXX
|
|
|
|
|
#self.calculateOffsets(self.selectedShape, pos)
|
|
|
|
|
dp = pos - self.prevPoint
|
|
|
|
|
if dp:
|
2021-02-05 12:05:11 +08:00
|
|
|
|
for shape in shapes:
|
|
|
|
|
shape.moveBy(dp)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.prevPoint = pos
|
|
|
|
|
return True
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
def deSelectShape(self):
|
2021-02-05 12:05:11 +08:00
|
|
|
|
if self.selectedShapes:
|
|
|
|
|
for shape in self.selectedShapes: shape.selected=False
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.setHiding(False)
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.selectionChanged.emit([])
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.update()
|
|
|
|
|
|
|
|
|
|
def deleteSelected(self):
|
2021-02-05 12:05:11 +08:00
|
|
|
|
deleted_shapes = []
|
|
|
|
|
if self.selectedShapes:
|
|
|
|
|
for shape in self.selectedShapes:
|
|
|
|
|
self.shapes.remove(shape)
|
|
|
|
|
deleted_shapes.append(shape)
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.storeShapes()
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.selectedShapes = []
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.update()
|
2021-02-05 12:05:11 +08:00
|
|
|
|
return deleted_shapes
|
|
|
|
|
|
|
|
|
|
def storeShapes(self):
|
|
|
|
|
shapesBackup = []
|
|
|
|
|
for shape in self.shapes:
|
|
|
|
|
shapesBackup.append(shape.copy())
|
|
|
|
|
if len(self.shapesBackups) >= 10:
|
|
|
|
|
self.shapesBackups = self.shapesBackups[-9:]
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.shapesBackups.append(shapesBackup)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
def copySelectedShape(self):
|
2021-02-05 12:05:11 +08:00
|
|
|
|
if self.selectedShapes:
|
|
|
|
|
self.selectedShapesCopy = [s.copy() for s in self.selectedShapes]
|
|
|
|
|
self.boundedShiftShapes(self.selectedShapesCopy)
|
|
|
|
|
self.endMove(copy=True)
|
|
|
|
|
return self.selectedShapes
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
2021-02-07 17:28:10 +08:00
|
|
|
|
def boundedShiftShapes(self, shapes):
|
2020-12-11 19:23:27 +08:00
|
|
|
|
# Try to move in one direction, and if it fails in another.
|
|
|
|
|
# Give up if both fail.
|
2021-02-07 17:28:10 +08:00
|
|
|
|
for shape in shapes:
|
|
|
|
|
point = shape[0]
|
|
|
|
|
offset = QPointF(2.0, 2.0)
|
|
|
|
|
self.calculateOffsets(shape, point)
|
|
|
|
|
self.prevPoint = point
|
|
|
|
|
if not self.boundedMoveShape(shape, point - offset):
|
|
|
|
|
self.boundedMoveShape(shape, point + offset)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
def paintEvent(self, event):
|
|
|
|
|
if not self.pixmap:
|
|
|
|
|
return super(Canvas, self).paintEvent(event)
|
|
|
|
|
|
|
|
|
|
p = self._painter
|
|
|
|
|
p.begin(self)
|
|
|
|
|
p.setRenderHint(QPainter.Antialiasing)
|
|
|
|
|
p.setRenderHint(QPainter.HighQualityAntialiasing)
|
|
|
|
|
p.setRenderHint(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)
|
2021-02-05 12:05:11 +08:00
|
|
|
|
if self.selectedShapesCopy:
|
|
|
|
|
for s in self.selectedShapesCopy:
|
|
|
|
|
s.paint(p)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
# Paint rect
|
|
|
|
|
if self.current is not None and len(self.line) == 2 and not self.fourpoint:
|
|
|
|
|
# print('Drawing rect')
|
|
|
|
|
leftTop = self.line[0]
|
|
|
|
|
rightBottom = self.line[1]
|
|
|
|
|
rectWidth = rightBottom.x() - leftTop.x()
|
|
|
|
|
rectHeight = rightBottom.y() - leftTop.y()
|
|
|
|
|
p.setPen(self.drawingRectColor)
|
|
|
|
|
brush = QBrush(Qt.BDiagPattern)
|
|
|
|
|
p.setBrush(brush)
|
|
|
|
|
p.drawRect(leftTop.x(), leftTop.y(), rectWidth, rectHeight)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ADD:
|
|
|
|
|
if (
|
|
|
|
|
self.fillDrawing()
|
|
|
|
|
and self.fourpoint
|
|
|
|
|
and self.current is not None
|
|
|
|
|
and len(self.current.points) >= 2
|
|
|
|
|
):
|
|
|
|
|
print('paint event')
|
|
|
|
|
drawing_shape = self.current.copy()
|
|
|
|
|
drawing_shape.addPoint(self.line[1])
|
|
|
|
|
drawing_shape.fill = True
|
|
|
|
|
drawing_shape.paint(p)
|
|
|
|
|
|
|
|
|
|
if self.drawing() and not self.prevPoint.isNull() and not self.outOfPixmap(self.prevPoint):
|
|
|
|
|
p.setPen(QColor(0, 0, 0))
|
|
|
|
|
p.drawLine(self.prevPoint.x(), 0, self.prevPoint.x(), self.pixmap.height())
|
|
|
|
|
p.drawLine(0, self.prevPoint.y(), self.pixmap.width(), self.prevPoint.y())
|
|
|
|
|
|
|
|
|
|
self.setAutoFillBackground(True)
|
|
|
|
|
if self.verified:
|
|
|
|
|
pal = self.palette()
|
|
|
|
|
pal.setColor(self.backgroundRole(), QColor(184, 239, 38, 128))
|
|
|
|
|
self.setPalette(pal)
|
|
|
|
|
else:
|
|
|
|
|
pal = self.palette()
|
|
|
|
|
pal.setColor(self.backgroundRole(), QColor(232, 232, 232, 255))
|
|
|
|
|
self.setPalette(pal)
|
|
|
|
|
|
|
|
|
|
p.end()
|
|
|
|
|
|
|
|
|
|
def fillDrawing(self):
|
|
|
|
|
return self._fill_drawing
|
|
|
|
|
|
|
|
|
|
def transformPos(self, point):
|
|
|
|
|
"""Convert from widget-logical coordinates to painter-logical coordinates."""
|
|
|
|
|
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
|
|
|
|
|
return 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
|
|
|
|
|
if self.current.points[0] == self.current.points[-1]:
|
|
|
|
|
# print('finalse')
|
|
|
|
|
self.current = None
|
|
|
|
|
self.drawingPolygon.emit(False)
|
|
|
|
|
self.update()
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
self.current.close()
|
|
|
|
|
self.shapes.append(self.current)
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
# 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):
|
|
|
|
|
qt_version = 4 if hasattr(ev, "delta") else 5
|
|
|
|
|
if qt_version == 4:
|
|
|
|
|
if ev.orientation() == Qt.Vertical:
|
|
|
|
|
v_delta = ev.delta()
|
|
|
|
|
h_delta = 0
|
|
|
|
|
else:
|
|
|
|
|
h_delta = ev.delta()
|
|
|
|
|
v_delta = 0
|
|
|
|
|
else:
|
|
|
|
|
delta = ev.angleDelta()
|
|
|
|
|
h_delta = delta.x()
|
|
|
|
|
v_delta = delta.y()
|
|
|
|
|
|
|
|
|
|
mods = ev.modifiers()
|
|
|
|
|
if Qt.ControlModifier == int(mods) and v_delta:
|
|
|
|
|
self.zoomRequest.emit(v_delta)
|
|
|
|
|
else:
|
|
|
|
|
v_delta and self.scrollRequest.emit(v_delta, Qt.Vertical)
|
|
|
|
|
h_delta and self.scrollRequest.emit(h_delta, Qt.Horizontal)
|
|
|
|
|
ev.accept()
|
|
|
|
|
|
|
|
|
|
def keyPressEvent(self, ev):
|
|
|
|
|
key = ev.key()
|
2021-07-28 21:12:00 +08:00
|
|
|
|
shapesBackup = []
|
|
|
|
|
shapesBackup = copy.deepcopy(self.shapes)
|
|
|
|
|
self.shapesBackups.pop()
|
|
|
|
|
self.shapesBackups.append(shapesBackup)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
if key == Qt.Key_Escape and self.current:
|
|
|
|
|
print('ESC press')
|
|
|
|
|
self.current = None
|
|
|
|
|
self.drawingPolygon.emit(False)
|
|
|
|
|
self.update()
|
|
|
|
|
elif key == Qt.Key_Return and self.canCloseShape():
|
|
|
|
|
self.finalise()
|
2021-07-22 14:49:31 +08:00
|
|
|
|
elif key == Qt.Key_Left and self.selectedShapes:
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.moveOnePixel('Left')
|
2021-07-22 14:49:31 +08:00
|
|
|
|
elif key == Qt.Key_Right and self.selectedShapes:
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.moveOnePixel('Right')
|
2021-07-22 14:49:31 +08:00
|
|
|
|
elif key == Qt.Key_Up and self.selectedShapes:
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.moveOnePixel('Up')
|
2021-07-22 14:49:31 +08:00
|
|
|
|
elif key == Qt.Key_Down and self.selectedShapes:
|
2021-02-07 17:28:10 +08:00
|
|
|
|
self.moveOnePixel('Down')
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
def moveOnePixel(self, direction):
|
|
|
|
|
# print(self.selectedShape.points)
|
2021-07-28 21:12:00 +08:00
|
|
|
|
self.selectCount = len(self.selectedShapes)
|
|
|
|
|
self.selectCountShape = True
|
2021-07-22 14:49:31 +08:00
|
|
|
|
for i in range(len(self.selectedShapes)):
|
|
|
|
|
self.selectedShape = self.selectedShapes[i]
|
|
|
|
|
if direction == 'Left' and not self.moveOutOfBound(QPointF(-1.0, 0)):
|
|
|
|
|
# print("move Left one pixel")
|
|
|
|
|
self.selectedShape.points[0] += QPointF(-1.0, 0)
|
|
|
|
|
self.selectedShape.points[1] += QPointF(-1.0, 0)
|
|
|
|
|
self.selectedShape.points[2] += QPointF(-1.0, 0)
|
|
|
|
|
self.selectedShape.points[3] += QPointF(-1.0, 0)
|
|
|
|
|
elif direction == 'Right' and not self.moveOutOfBound(QPointF(1.0, 0)):
|
|
|
|
|
# print("move Right one pixel")
|
|
|
|
|
self.selectedShape.points[0] += QPointF(1.0, 0)
|
|
|
|
|
self.selectedShape.points[1] += QPointF(1.0, 0)
|
|
|
|
|
self.selectedShape.points[2] += QPointF(1.0, 0)
|
|
|
|
|
self.selectedShape.points[3] += QPointF(1.0, 0)
|
|
|
|
|
elif direction == 'Up' and not self.moveOutOfBound(QPointF(0, -1.0)):
|
|
|
|
|
# print("move Up one pixel")
|
|
|
|
|
self.selectedShape.points[0] += QPointF(0, -1.0)
|
|
|
|
|
self.selectedShape.points[1] += QPointF(0, -1.0)
|
|
|
|
|
self.selectedShape.points[2] += QPointF(0, -1.0)
|
|
|
|
|
self.selectedShape.points[3] += QPointF(0, -1.0)
|
|
|
|
|
elif direction == 'Down' and not self.moveOutOfBound(QPointF(0, 1.0)):
|
|
|
|
|
# print("move Down one pixel")
|
|
|
|
|
self.selectedShape.points[0] += QPointF(0, 1.0)
|
|
|
|
|
self.selectedShape.points[1] += QPointF(0, 1.0)
|
|
|
|
|
self.selectedShape.points[2] += QPointF(0, 1.0)
|
|
|
|
|
self.selectedShape.points[3] += QPointF(0, 1.0)
|
2021-08-11 11:34:08 +08:00
|
|
|
|
shapesBackup = []
|
|
|
|
|
shapesBackup = copy.deepcopy(self.shapes)
|
|
|
|
|
self.shapesBackups.append(shapesBackup)
|
|
|
|
|
self.shapeMoved.emit()
|
|
|
|
|
self.repaint()
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
def moveOutOfBound(self, step):
|
|
|
|
|
points = [p1+p2 for p1, p2 in zip(self.selectedShape.points, [step]*4)]
|
|
|
|
|
return True in map(self.outOfPixmap, points)
|
|
|
|
|
|
|
|
|
|
def setLastLabel(self, text, line_color = None, fill_color = None):
|
|
|
|
|
assert text
|
|
|
|
|
self.shapes[-1].label = text
|
|
|
|
|
if line_color:
|
|
|
|
|
self.shapes[-1].line_color = line_color
|
|
|
|
|
|
|
|
|
|
if fill_color:
|
|
|
|
|
self.shapes[-1].fill_color = fill_color
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.storeShapes()
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
|
2021-02-05 12:05:11 +08:00
|
|
|
|
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()
|
|
|
|
|
|
2020-12-11 19:23:27 +08:00
|
|
|
|
def resetAllLines(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)
|
|
|
|
|
self.current = None
|
|
|
|
|
self.drawingPolygon.emit(False)
|
|
|
|
|
self.update()
|
|
|
|
|
|
|
|
|
|
def loadPixmap(self, pixmap):
|
|
|
|
|
self.pixmap = pixmap
|
|
|
|
|
self.shapes = []
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.repaint()
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
2021-02-05 12:05:11 +08:00
|
|
|
|
def loadShapes(self, shapes, replace=True):
|
|
|
|
|
if replace:
|
|
|
|
|
self.shapes = list(shapes)
|
|
|
|
|
else:
|
|
|
|
|
self.shapes.extend(shapes)
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.current = None
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.hShape = None
|
|
|
|
|
self.hVertex = None
|
|
|
|
|
# self.hEdge = None
|
|
|
|
|
self.storeShapes()
|
2020-12-11 19:23:27 +08:00
|
|
|
|
self.repaint()
|
|
|
|
|
|
|
|
|
|
def setShapeVisible(self, shape, value):
|
|
|
|
|
self.visible[shape] = value
|
|
|
|
|
self.repaint()
|
|
|
|
|
|
|
|
|
|
def currentCursor(self):
|
|
|
|
|
cursor = QApplication.overrideCursor()
|
|
|
|
|
if cursor is not None:
|
|
|
|
|
cursor = cursor.shape()
|
|
|
|
|
return cursor
|
|
|
|
|
|
|
|
|
|
def overrideCursor(self, cursor):
|
|
|
|
|
self._cursor = cursor
|
|
|
|
|
if self.currentCursor() is None:
|
|
|
|
|
QApplication.setOverrideCursor(cursor)
|
|
|
|
|
else:
|
|
|
|
|
QApplication.changeOverrideCursor(cursor)
|
|
|
|
|
|
|
|
|
|
def restoreCursor(self):
|
|
|
|
|
QApplication.restoreOverrideCursor()
|
|
|
|
|
|
|
|
|
|
def resetState(self):
|
|
|
|
|
self.restoreCursor()
|
|
|
|
|
self.pixmap = None
|
|
|
|
|
self.update()
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.shapesBackups = []
|
2020-12-11 19:23:27 +08:00
|
|
|
|
|
|
|
|
|
def setDrawingShapeToSquare(self, status):
|
|
|
|
|
self.drawSquare = status
|
2021-02-05 12:05:11 +08:00
|
|
|
|
|
2021-02-07 17:28:10 +08:00
|
|
|
|
def restoreShape(self):
|
2021-02-05 12:05:11 +08:00
|
|
|
|
if not self.isShapeRestorable:
|
|
|
|
|
return
|
2021-08-11 11:34:08 +08:00
|
|
|
|
|
2021-02-05 12:05:11 +08:00
|
|
|
|
self.shapesBackups.pop() # latest
|
|
|
|
|
shapesBackup = self.shapesBackups.pop()
|
|
|
|
|
self.shapes = shapesBackup
|
|
|
|
|
self.selectedShapes = []
|
|
|
|
|
for shape in self.shapes:
|
|
|
|
|
shape.selected = False
|
|
|
|
|
self.repaint()
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def isShapeRestorable(self):
|
|
|
|
|
if len(self.shapesBackups) < 2:
|
|
|
|
|
return False
|
|
|
|
|
return True
|