QPainter 在 QScrollArea 内画线

QPainter draw lines inside a QScrollArea

我之前提出了一个关于如何在小部件之间画线的问题。 @eyllanesc 很好地解决了这个问题。但是我遇到了另一个问题。我尝试用 QScrollarea 制作小部件。但是没有显示线条。

当使用 Widget 时,它可以很好地绘制线条,但 QScrollArea 似乎需要另一个 paint_event。不知道该怎么做。

谁能帮我用QScrollarea显示线条。

import functools
import sys

from PyQt5.QtCore import QEvent, QLine
from PyQt5.QtGui import QPainter
from PyQt5.QtWidgets import QApplication, QPushButton, QVBoxLayout, QWidget, QScrollArea


class Drawer:
    def __init__(self):
        self._lines = []

    @property
    def lines(self):
        return self._lines

    @lines.setter
    def lines(self, l):
        self._lines = l[:]
        self.update()

    def paintEvent(self, event):
        painter = QPainter(self)
        for line in self.lines:
            painter.drawLine(line)


class Example(QScrollArea, Drawer):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.initUI()

    def initUI(self):
        self.AddButton = QPushButton("Add")
        self.AddButton.clicked.connect(self.addbuttonwithline)

        self.w = QWidget()
        self.setWidget(self.w)
        self.setWidgetResizable(True)

        self.vbox = QVBoxLayout(self.w)
        self.vbox.addWidget(self.AddButton)

        self.setGeometry(300, 300, 300, 150)
        self.setWindowTitle("Buttons")

        self.buttons = []

    def addbuttonwithline(self):
        button = QPushButton("delete")
        button.clicked.connect(functools.partial(self.remove_button, button))
        button.installEventFilter(self)
        self.vbox.addWidget(button)
        self.buttons.append(button)
        self.recalculate_position()

    def remove_button(self, button):
        self.buttons.remove(button)
        button.deleteLater()
        self.recalculate_position()

    def recalculate_position(self):
        lines = []
        for last_button, next_button in zip(self.buttons, self.buttons[1:]):
            l = QLine(last_button.pos().x()+50, last_button.pos().y(), next_button.pos().x()+50, next_button.pos().y())
            lines.append(l)
        self.lines = lines

    def eventFilter(self, o, e):
        if e.type() == QEvent.Move and o in self.buttons:
            self.recalculate_position()
        return super().eventFilter(o, e)


if __name__ == "__main__":

    app = QApplication(sys.argv)
    ex = Example()
    ex.show()
    sys.exit(app.exec_())

QScrollArea 被设计成一个容器,因此您不应在该项目上绘画,因为内容覆盖了它,而是在内容小部件中设置绘画。

根据我在 中的代码,然后只需更改为:

if __name__ == "__main__":

    app = QApplication(sys.argv)
    ex = Example()
    w = QScrollArea(widgetResizable=True)
    w.setWidget(ex)
    w.show()
    sys.exit(app.exec_())