【问题标题】:How to paint QGraphicsItems from scene onto QImage without changing them in the QGraphicsScene?如何将场景中的 QGraphicsItems 绘制到 QImage 上而不在 QGraphicsScene 中更改它们?
【发布时间】:2020-06-18 09:21:24
【问题描述】:

我正在将QGraphicsItem 转换为QGraphicsScene 中它们所在位置的光栅化蒙版。我正在使用这些掩码对视频进行进一步处理(取掩码内的平均强度)。为了实现这一点,我在QImage 上一个一个地绘制场景中的每个项目,它的大小刚好可以包围项目。一切运行良好,但场景中的项目消失了。那是因为当我在QImage 上绘画时,我将笔从项目中取出。完成后我将原来的笔放回原处,但这些物品没有重新出现在现场。 如何“刷新”场景以使项目重新出现,或者防止项目完全消失?

我真的找不到任何遇到这个问题的人。所以也许我只是在做一些根本错误的事情。欢迎提出任何建议。

这是我的代码:

class MyThread(QtCore.QThread):
    def __init__(self, scene):
        super().__init__()
        self.scene = scene

    def run(self):
        for item in self.scene.items():
            # Render the ROI item to create a rasterized mask.
            qimage = self.qimage_from_shape_item(item)
            # do some stuff

    @staticmethod
    def qimage_from_shape_item(item: QtWidgets.QAbstractGraphicsShapeItem) -> QtGui.QImage:
        # Get items pen and brush to set back later.
        pen = item.pen()
        brush = item.brush()
        # Remove pen, set brush to white.
        item.setPen(QtGui.QPen(QtCore.Qt.NoPen))
        item.setBrush(QtCore.Qt.white)

        # Determine the bounding box in pixel coordinates.
        top = int(item.scenePos().y() + item.boundingRect().top())
        left = int(item.scenePos().x() + item.boundingRect().left())
        bottom = int(item.scenePos().y() + item.boundingRect().bottom()) + 1
        right = int(item.scenePos().x() + item.boundingRect().right()) + 1

        size = QtCore.QSize(right - left, bottom - top)

        # Initialize qimage, use 8-bit grayscale.
        qimage = QtGui.QImage(size, QtGui.QImage.Format_Grayscale8)
        qimage.fill(QtCore.Qt.transparent)
        painter = QtGui.QPainter(qimage)

        painter.setRenderHint(QtGui.QPainter.Antialiasing)

        # Offset the painter to paint item in its correct pixel location.
        painter.translate(item.scenePos().x() - left, item.scenePos().y() - top)

        # Paint the item.
        item.paint(painter, QtWidgets.QStyleOptionGraphicsItem())

        # Set the pen and brush back.
        item.setPen(pen)
        item.setBrush(brush)

        # Set the pixel coordinate offset of the item to the QImage.
        qimage.setOffset(QtCore.QPoint(left, top))

        return qimage


if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)

    widget = QtWidgets.QWidget()
    layout = QtWidgets.QVBoxLayout(widget)

    view = QtWidgets.QGraphicsView()
    layout.addWidget(view)

    scene = QtWidgets.QGraphicsScene()
    view.setScene(scene)
    thread = MyThread(scene)

    view.setFixedSize(400, 300)
    scene.setSceneRect(0, 0, 400, 300)

    rect_item = QtWidgets.QGraphicsRectItem()

    p = QtCore.QPointF(123.4, 56.78)
    rect_item.setPos(p)

    r = QtCore.QRectF(0., 0., 161.8, 100.)
    rect_item.setRect(r)

    scene.addItem(rect_item)

    button = QtWidgets.QPushButton("Get masks")
    layout.addWidget(button)

    button.clicked.connect(thread.start)

    widget.show()

    sys.exit(app.exec_())

【问题讨论】:

    标签: python qt pyqt5 qgraphicsscene qgraphicsitem


    【解决方案1】:

    问题是您“只能在主线程上创建和使用 GUI 小部件”,请参阅this SO answer here 中的更多信息。

    我解决它的方法是将GUI交互部分,即qimage_from_shape_item()从线程中取出并在主循环中处理它。我想我直接使用这些项目仍然不是很好,虽然暂时设置 NoPen 没有可见的闪烁效果或任何东西。

    另一种可能是使用QGraphicsScene::render;但是,我不知道如何在不与场景中的其他项目交互的情况下逐个渲染项目。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2012-04-08
      • 2013-06-26
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2020-10-17
      • 1970-01-01
      • 2015-11-02
      相关资源
      最近更新 更多