PyQt5 объект группы художников

#python #python-3.x #pyqt #pyqt5 #qgraphicsview

#python #python-3.x #pyqt #pyqt5 #qgraphicsview

Вопрос:

Мне было интересно, есть ли способ нарисовать кучу фрагментов как свой собственный объект, чтобы, когда пользователь захочет масштабировать, или изменять цвета, или поворачивать все фигуры, они соответствующим образом корректировались?

Затем я хотел бы, чтобы эта группа фигур была размещена QGraphicsView в заданных координатах x, y с заданным поворотом. В настоящее время я получаю несколько ошибок, в которых говорится QPainter::scale: Painter not active и NotImplementedError: QGraphicsItem.paint() is abstract and must be overridden .

Я довольно новичок в графике и рисовании в PyQt и не уверен, что это правильный способ достижения того, чего я хотел бы. Любая помощь приветствуется

Код

 from PyQt5.QtCore import *
from PyQt5.QtGui import *
from PyQt5.QtWidgets import *
import sys
from math import sqrt

class LogObject(QObject):
    hovered = pyqtSignal()
    notHovered = pyqtSignal()

class ExamplePaintObject(QPainter):
    def __init__(self):
        super(ExamplePaintObject,self).__init__()
        self.color = QColor(0,0,0)

    def paintEvent(self,event):
        self.drawEllipse(0,0,0.0268,0.0268)
        self.drawEllipse(0,0,0.0357,0.0357)
        self.drawLine(0.0357,0.0045,0.0357 0.0821,0.0045)
        self.drawLine(0.0357,-0.0045,0.0357 0.0821,-0.0045)
        self.drawRect(0.0357 0.0821,-0.0045,0.1450,0.1450)


class Point(QGraphicsItem):

    def __init__(self, x, y,r, name):

        super(Point,self).__init__()
        self.setFlag(QGraphicsItem.ItemIsSelectable, True)
        self.name = name

        self.ex= ExamplePaintObject()
        self.ex.scale(40,40)

        self.x = x
        self.y = y
        self.rotate = r
        self.setRotation(self.rotate)
        self.setAcceptHoverEvents(True)
        self.log = LogObject()
        self.setPos(x, y)
        self.isSelected = False


    def itemChange(self, change, value):
        if change == self.ItemSelectedChange:
            self.setBrush(QBrush(Qt.green) if value else QBrush(Qt.black))
        return QGraphicsItem.itemChange(self, change, value)

    def hoverEnterEvent(self, event):
        self.setBrush(QColor("red"))
        self.log.hovered.emit()
        QGraphicsItem.hoverMoveEvent(self, event)

    def hoverLeaveEvent(self, event):
        if self.isSelected == True:
            self.setBrush(QBrush(Qt.green))
        else:
            self.setBrush(QColor("black"))
        self.log.notHovered.emit()
        QGraphicsItem.hoverMoveEvent(self, event)

    def mousePressEvent(self, event):
        print(self.name)
        QGraphicsItem.mousePressEvent(self, event)


class Viewer(QGraphicsView):
    photoClicked = pyqtSignal(QPoint)
    rectChanged = pyqtSignal(QRect)

    def __init__(self, parent):
        super(Viewer, self).__init__(parent)
        self.rubberBand = QRubberBand(QRubberBand.Rectangle, self)
        self.setMouseTracking(True)
        self.origin = QPoint()
        self.changeRubberBand = False

        self._zoom = 0
        self._empty = True
        self.setScene(QGraphicsScene(self))

        self.setTransformationAnchor(QGraphicsView.AnchorUnderMouse)
        self.setResizeAnchor(QGraphicsView.AnchorUnderMouse)
        self.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
        self.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
        self.setFrameShape(QFrame.NoFrame)
        self.area = float()
        self.setPoints()
        self.viewport().setCursor(Qt.ArrowCursor)
        QTimer.singleShot(0, self.reset_fit)
        self.selectedItems = []

    def setItems(self):
        self.data = {
            "x": [
                -2414943.8686,
                -2417160.6592,
                -2417160.6592,
                -2417856.1783,
                -2417054.7618,
                -2416009.9966,
                -2416012.5232,
                -2418160.8952,
                -2418160.8952,
                -2416012.5232,
                -2417094.7694,
                -2417094.7694,
            ],
            "y": [
                10454269.7008,
                10454147.2672,
                10454147.2672,
                10453285.2456,
                10452556.8132,
                10453240.2808,
                10455255.8752,
                10455183.1912,
                10455183.1912,
                10455255.8752,
                10456212.5959,
                10456212.5959,
            ],
            "rotation":[
            313.9962,
            43.9962,
            223.9962,
            227.7070,
            227.7070,
            313.9962,
            43.9962,
            43.9962,
            223.9962,
            223.9962,
            43.9962,
            223.9962,
            ]
        }
        maxX = max(self.data["x"])
        minX = min(self.data["x"])
        maxY = max(self.data["y"])
        minY = min(self.data["y"])
        distance = sqrt((maxX - minX) ** 2   (maxY - minY) ** 2)

        self.area = QRectF(minX, minY, distance, distance)
        for i, (x, y,r) in enumerate(zip(self.data["x"], self.data["y"],self.data["rotation"])):
            p = Point(x, y,r, "Point__"   str(i))
            p.log.hovered.connect(self.hoverChange)
            p.log.notHovered.connect(self.notHoverChange)
            self.scene().addItem(p)

    def setPoints(self):
        self.setItems()
        self.setDragMode(self.ScrollHandDrag)

    def wheelEvent(self, event):
        if event.angleDelta().y() > 0:
            factor = 1.25
            self._zoom  = 1
        else:
            factor = 0.8
            self._zoom -= 1
        if self._zoom > 0:
            self.scale(factor, factor)
        elif self._zoom == 0:
            self.reset_fit()
        else:
            self._zoom = 0

    def hoverChange(self):
        self.viewport().setCursor(Qt.PointingHandCursor)

    def notHoverChange(self):
        self.viewport().setCursor(Qt.ArrowCursor)

    def mousePressEvent(self, event):
        if event.button() == Qt.LeftButton:
            singleItem = self.itemAt(event.pos().x(), event.pos().y())
            if singleItem != None:
                if QApplication.keyboardModifiers() == Qt.ShiftModifier: # This will determine if the shift key is depressed
                    if singleItem.isSelected == True:
                        singleItem.setSelected(False)
                        singleItem.isSelected = False
                        self.selectedItems.remove(singleItem)
                elif singleItem.isSelected == False:
                    singleItem.setSelected(True)
                    singleItem.isSelected = True
                    self.selectedItems.append(singleItem)
            else:
                self.origin = event.pos()
                self.rubberBand.setGeometry(QRect(self.origin, QSize()))
                self.rectChanged.emit(self.rubberBand.geometry())
                self.rubberBand.show()
                self.changeRubberBand = True
            return

        elif event.button() == Qt.MidButton:
            self.viewport().setCursor(Qt.ClosedHandCursor)
            self.original_event = event
            handmade_event = QMouseEvent(
                QEvent.MouseButtonPress,
                QPointF(event.pos()),
                Qt.LeftButton,
                event.buttons(),
                Qt.KeyboardModifiers(),
            )
            QGraphicsView.mousePressEvent(self, handmade_event)

        super(Viewer, self).mousePressEvent(event)

    def mouseReleaseEvent(self, event):
        point = event.pos()
        # print(self.mapToScene(point))
        if event.button() == Qt.LeftButton:
            self.changeRubberBand = False
            if self.rubberBand.isVisible():
                self.rubberBand.hide()
                rect = self.rubberBand.geometry()
                rect_scene = self.mapToScene(rect).boundingRect()
                selected = self.scene().items(rect_scene)
                if selected:

                    for selectedPoints in selected:
                        if QApplication.keyboardModifiers() == Qt.ShiftModifier: # This will determine if the shift key is depressed
                            if selectedPoints.isSelected == True:
                                selectedPoints.setSelected(False)
                                selectedPoints.isSelected = False
                                self.selectedItems.remove(selectedPoints)
                        elif selectedPoints.isSelected == False: # if the shif key is not depressed and its not selected, then select it
                            selectedPoints.setSelected(True)
                            selectedPoints.isSelected = True
                            self.selectedItems.append(selectedPoints)
                    print( "".join("Item: %sn" % child.name for child in self.selectedItems))
                else:
                    print(" Nothingn")
                    for selected in self.selectedItems:
                        selected.setSelected(False)
                        selected.isSelected = False
                    self.selectedItems.clear()
                    QGraphicsView.mouseReleaseEvent(self, event)

        elif event.button() == Qt.MidButton:
            self.viewport().setCursor(Qt.ArrowCursor)
            handmade_event = QMouseEvent(
                QEvent.MouseButtonRelease,
                QPointF(event.pos()),
                Qt.LeftButton,
                event.buttons(),
                Qt.KeyboardModifiers(),
            )
            QGraphicsView.mouseReleaseEvent(self, handmade_event)
        # super(Viewer, self).mouseReleaseEvent(event)

    def mouseMoveEvent(self, event):
        if self.changeRubberBand:
            self.rubberBand.setGeometry(
                QRect(self.origin, event.pos()).normalized()
            )
            self.rectChanged.emit(self.rubberBand.geometry())
            QGraphicsView.mouseMoveEvent(self, event)
        super(Viewer, self).mouseMoveEvent(event)

    def reset_fit(self):
        r = self.scene().itemsBoundingRect()
        self.resetTransform()
        self.setSceneRect(r)
        self.fitInView(r, Qt.KeepAspectRatio)
        self._zoom = 0
        self.scale(1, -1)


class Window(QWidget):
    def __init__(self):
        super(Window, self).__init__()
        self.viewer = Viewer(self)
        self.btnLoad = QToolButton(self)
        self.btnLoad.setText("Fit Into View")
        self.btnLoad.clicked.connect(self.fitPoints)

        VBlayout = QVBoxLayout(self)
        VBlayout.addWidget(self.viewer)
        HBlayout = QHBoxLayout()
        HBlayout.setAlignment(Qt.AlignLeft)
        HBlayout.addWidget(self.btnLoad)

        VBlayout.addLayout(HBlayout)

    def fitPoints(self):
        self.viewer.reset_fit()

if __name__ == '__main__':
    import sys
    app = QApplication(sys.argv)
    window = Window()
    window.setGeometry(500, 300, 800, 600)
    window.show()
    sys.exit(app.exec_())
 

Комментарии:

1. вы могли бы объяснить мне лучше, я не понимаю, что вы хотите сделать

2. @eyllanesc Я хотел бы нарисовать несколько пользовательских значков, а затем разместить их в этих местах X и Y. В итоге у меня будет около 20 разных значков, и я подумал, что если я сделаю каждый значок классом, то конечный пользователь сможет выбрать, какой значок он хотел бы использовать. Кроме того, я хотел бы, чтобы значки могли масштабироваться, поворачиваться и изменять цвета. Я надеюсь, что я достаточно хорошо это объясняю

3. но в вашем вопросе я не вижу значков.

4. ExamplePaintObject Класс — это то, что я пытался использовать для создания значка

5. хорошо, но почему такие маленькие координаты: 0,0,0.0268,0.0268 ?

Ответ №1:

Возможным решением является использование QPainterPath:

 def create_path():
    path = QPainterPath()
    path.addEllipse(0, 0, 0.0268, 0.0268)
    path.addEllipse(0, 0, 0.0357, 0.0357)
    path.moveTo(0.0357, 0.0045)
    path.lineTo(0.0357   0.0821, 0.0045)
    path.moveTo(0.0357, -0.0045)
    path.lineTo(0.0357   0.0821, -0.0045)
    path.addRect(0.0357   0.0821, -0.0045, 0.1450, 0.1450)
    tr = QTransform()
    tr.scale(100, 100)
    path = tr.map(path)
    return path


class Point(QGraphicsPathItem):
    def __init__(self, x, y, r, name):
        super(Point, self).__init__()
        self.setFlag(QGraphicsItem.ItemIsSelectable, True)
        self.name = name
        self.setPath(create_path())
        self.setScale(10)
        # ...
 

Для лучшего наблюдения вы должны использовать setRenderHints (QPainter.Сглаживание) в QGraphicView:

 class Viewer(QGraphicsView):
    photoClicked = pyqtSignal(QPoint)
    rectChanged = pyqtSignal(QRect)

    def __init__(self, parent):
        super(Viewer, self).__init__(parent)
        self.rubberBand = QRubberBand(QRubberBand.Rectangle, self)
        self.setMouseTracking(True)
        self.origin = QPoint()
        self.changeRubberBand = False
        self.setRenderHints(QPainter.Antialiasing) # <---
        # ....
 

введите описание изображения здесь

Комментарии:

1. Это здорово! Вы получили именно то, о чем я спрашивал. Спасибо