我正在尝试覆盖2个像素图,并将它们转换为QGraphics场景中的单个像素图。这两个像素图在某些位置都是透明的。我想使用here列出的'SourceOver'混合类型来组合这些地图:下面有一个简单的玩具示例来说明我创建了两个虚拟的透明像素图(一个绿色和一个蓝色)的问题。实际上,这些地图是从图像加载并绘制的,但是此示例重现了该问题。基于此How to add an image on the top of another image?,我尝试的方法(注释掉了4行)是使用一个像素图创建一个QPainter,然后在其上方绘制另一个像素图,但是这会使程序崩溃。有想法该怎么解决这个吗?我最终希望能够保存合并的像素图。

import sys
from PyQt5 import QtCore, QtGui, QtWidgets
from PyQt5.QtGui import QPixmap, QPainter, QPen, QBrush, QPainterPath
from PyQt5.QtCore import (QLineF, QPointF, QRectF, Qt)

class Viewer(QtWidgets.QGraphicsView):
    def __init__(self, parent):
        super(Viewer, self).__init__(parent)
        self._scene = QtWidgets.QGraphicsScene(self)
        self.photo = QtWidgets.QGraphicsPixmapItem()
        self.label = QtWidgets.QGraphicsPixmapItem()
        self._scene.addItem(self.photo)
        self._scene.addItem(self.label)
        self.setScene(self._scene)

    def overlayMaps(self):
        blue = QtGui.QPixmap(600, 600)
        blue.fill(QtGui.QColor(0,0,255,0))
        p = QPainter(blue)
        self.pen = QPen()
        self.pen.setColor(QtGui.QColor(0,0,255,255))
        self.pen.setWidth(10)
        p.setPen(self.pen)
        p.drawLine(0,0,600,600)


        green = QtGui.QPixmap(600, 600)
        green.fill(QtGui.QColor(0,255,0,0))
        p = QPainter(green)
        self.pen = QPen()
        self.pen.setColor(QtGui.QColor(0,255,0,255))
        self.pen.setWidth(10)
        p.setPen(self.pen)
        p.drawLine(600,0,0,600)

        self.photo.setPixmap(blue)
        self.label.setPixmap(green)

        resultPixmap = QtGui.QPixmap(self.photo.pixmap().width(), self.photo.pixmap().height())
#        resultPainter = QtGui.QPainter(resultPixmap)
#        resultPainter.setCompositionMode(QtGui.QPainter.CompositionMode_SourceOver)
#        resultPainter.drawPixmap(300,300, self.photo.pixmap())
#        resultPainter.drawPixmap(300,300, self.label.pixmap())


    def saveOverlayMap(self):
        pass

class Window(QtWidgets.QWidget):
    def __init__(self):
        super(Window, self).__init__()
        self.viewer = Viewer(self)
        self.viewer.overlayMaps()

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

最佳答案

我实现了一个功能,该功能根据模式执行加入的动作,为了更好的理解,我已经移动了这些项目。

import sys
from PyQt5 import QtCore, QtGui, QtWidgets

def join_pixmap(p1, p2, mode=QtGui.QPainter.CompositionMode_SourceOver):
    s = p1.size().expandedTo(p2.size())
    result =  QtGui.QPixmap(s)
    result.fill(QtCore.Qt.transparent)
    painter = QtGui.QPainter(result)
    painter.setRenderHint(QtGui.QPainter.Antialiasing)
    painter.drawPixmap(QtCore.QPoint(), p1)
    painter.setCompositionMode(mode)
    painter.drawPixmap(result.rect(), p2, p2.rect())
    painter.end()
    return result

class Viewer(QtWidgets.QGraphicsView):
    def __init__(self, parent=None):
        super(Viewer, self).__init__(parent)
        self._scene = QtWidgets.QGraphicsScene(self)
        self.setScene(self._scene)

        blue = QtGui.QPixmap(100, 100)
        blue.fill(QtCore.Qt.transparent)
        p = QtGui.QPainter(blue)
        pen = QtGui.QPen(QtGui.QBrush(QtGui.QColor(0,0,255)), 10)
        p.setPen(pen)
        p.drawLine(0, 0, 100, 100)
        p.end()
        self.photo = self._scene.addPixmap(blue)

        green = QtGui.QPixmap(100, 100)
        green.fill(QtCore.Qt.transparent)
        p = QtGui.QPainter(green)
        pen = QtGui.QPen(QtGui.QBrush(QtGui.QColor(0, 255, 0, 255)), 10)
        p.setPen(pen)
        p.drawLine(100, 0, 0, 100)
        p.end()
        self.label = self._scene.addPixmap(green)
        self.label.setPos(200, 0)

        self.overlayMaps()

    def overlayMaps(self):
        p1 = QtGui.QPixmap(self.photo.pixmap())
        p2 = QtGui.QPixmap(self.label.pixmap())

        result_pixmap = join_pixmap(self.photo.pixmap(), self.label.pixmap())
        self.result_item = self._scene.addPixmap(result_pixmap)
        self.result_item.setPos(100, 200)

        result_pixmap.save("result_pixmap.png")

if __name__ == '__main__':
    import sys
    app = QtWidgets.QApplication(sys.argv)
    window = Viewer()
    window.resize(640, 480)
    window.show()
    sys.exit(app.exec_())


python - 使用QPainter用alpha值覆盖两个像素图-LMLPHP

result_pixmap.png

python - 使用QPainter用alpha值覆盖两个像素图-LMLPHP

关于python - 使用QPainter用alpha值覆盖两个像素图,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/53420826/

10-10 14:00