pyqt5用QGraphicsScene上的按钮绘制线条

时间:2020-05-04 16:35:12

标签: python qt pyqt5 qgraphicsview qgraphicsscene

我对pyqt5有问题。我创建了一个带有背景图片的场景窗口,重新实现了drawBackground。我还有一个按钮,可让我在场景中的某个位置添加一条线。问题是,如果我单击按钮绘制线条,则此线条将在具有其自身背景的单独场景中绘制,而不是在我所拥有的场景中绘制。好像它创建了一个新的场景来画线。这是我的代码:

import sys

from PyQt5 import QtGui
from PyQt5.QtGui import QImage
from PyQt5.QtWidgets import (QMainWindow, QGraphicsView, QPushButton, 
    QHBoxLayout, QVBoxLayout, QWidget, QApplication, QGraphicsScene)

class GraphicsScene(QGraphicsScene):

  def __init__(self, *args, **kwargs):
    super().__init__(*args, **kwargs)
    self._image = QImage()

  @property
  def image(self):
    return self._image

  @image.setter
  def image(self, img):
    self._image = img
    self.update()

  def drawBackground(self, painter, rect):
    if self.image.isNull():
      super().drawBackground(painter, rect)
    else:
      painter.drawImage(rect, self._image)

class MainWindow(QWidget):
    def __init__(self):
        super().__init__()

        self.title = "parcelDeliveryIta";
        self.top = 100
        self.left = 100
        self.width = 1500
        self.height = 900
        self.initUI()

    def initUI(self):

        self.scene = GraphicsScene(self)
        self.scene._image = QImage('Italy.png')
        view = QGraphicsView(self.scene, self)
        self.scene.setSceneRect(0, 0, view.width(), view.height())

        addLine = QPushButton('AddLine')
        addLine.clicked.connect(self.addLine)

        hbox = QHBoxLayout(self)
        hbox.addWidget(view)

        vbox = QVBoxLayout(self)
        vbox.addWidget(addLine)

        hbox.addLayout(vbox)

        self.setWindowTitle(self.title)
        self.setGeometry(self.top, self.left, self.width, self.height)
        self.setFixedSize(self.width, self.height)
        self.setLayout(hbox)
        self.show()

    def addLine(self):
        self.scene.addLine(0, 0, 100, 100)


if __name__ == "__main__":

    app = QApplication(sys.argv)
    window = MainWindow()
    sys.exit(app.exec_())

这是单击按钮时的结果:

enter image description here

可以看到,线条是用自己的背景画的,即我将背景图像设置为场景的背景(上面的图像经过裁剪以更好地显示线条) 感谢您的帮助。

1 个答案:

答案 0 :(得分:0)

说明:

似乎my previous solution不适合您的情况,因为the docs指出:

void QGraphicsScene::drawBackground(QPainter *painter, const QRectF &rect)

使用painter绘制场景的背景, 在绘制任何项目之前和前景。重新实现 函数为场景提供自定义背景。

所有绘制都在场景坐标中完成。 rect参数是 裸露的矩形。

如果您只想为颜色定义颜色,纹理或渐变 背景,您可以改为调用setBackgroundBrush()。

另请参阅drawForeground()和drawItems()。

(重点是我的)

该油漆也将用于油漆物品的底部,从而导致该行为。


解决方案:

因此,您将不得不求助于另一种解决方案,例如以QGraphicsPixmapItem为基础,并使用该信息重新调整窗口的大小:

import sys

from PyQt5.QtGui import QPixmap
from PyQt5.QtWidgets import (
    QGraphicsView,
    QPushButton,
    QHBoxLayout,
    QVBoxLayout,
    QWidget,
    QApplication,
    QGraphicsScene,
)


class GraphicsView(QGraphicsView):
    def __init__(self, parent=None):
        super().__init__(parent)
        scene = QGraphicsScene(self)
        self.setScene(scene)
        self._pixmap_item = self.scene().addPixmap(QPixmap())
        self._pixmap_item.setZValue(-1)

    @property
    def pixmap(self):
        return self._pixmap_item.pixmap()

    @pixmap.setter
    def pixmap(self, pixmap):
        self._pixmap_item.setPixmap(pixmap)
        self.scene().setSceneRect(self._pixmap_item.boundingRect())

    def resizeEvent(self, event):
        if not self._pixmap_item.pixmap().isNull():
            self.fitInView(self._pixmap_item)
        super().resizeEvent(event)


class MainWindow(QWidget):
    def __init__(self):
        super().__init__()

        self.title = "parcelDeliveryIta"
        self.top = 100
        self.left = 100
        self.width = 1500
        self.height = 900
        self.initUI()

    def initUI(self):

        self.view = GraphicsView(self)
        self.view.pixmap = QPixmap("Italy.png")

        addLine = QPushButton("AddLine")
        addLine.clicked.connect(self.addLine)

        hbox = QHBoxLayout(self)
        hbox.addWidget(self.view)

        vbox = QVBoxLayout()
        vbox.addWidget(addLine)

        hbox.addLayout(vbox)

        self.setWindowTitle(self.title)
        self.setGeometry(self.top, self.left, self.width, self.height)
        self.setFixedSize(self.width, self.height)
        self.show()

    def addLine(self):
        self.view.scene().addLine(0, 0, 100, 100)


if __name__ == "__main__":

    app = QApplication(sys.argv)
    window = MainWindow()
    sys.exit(app.exec_())