首页 > 解决方案 > QPainter中重叠的半透明QRectangles

问题描述

当两个矩形重叠时,我注意到两者之间的交集变得更加不透明,据我所知,这是由于添加了 QBrush alphas。

重叠交叉点变得更加不透明的视觉示例:

重叠交叉点变得更加不透明的视觉示例

我想防止重叠的交叉点变得更加不透明,该区域应该与其余矩形具有相同的颜色。(使用完全不透明的 QColors 时会发生这种情况,但我想使用半透明的 QColors)。

我已经阅读了有关该主题的几个主题(包括这个非常相似的问题:Qt: Overlapping semitransparent QgraphicsItem),每个答案都表明可以通过更改画家的合成模式来解决问题,但是我就是找不到合成提供预期结果的模式,我已经尝试了 Qpainter 文档 ( https://doc.qt.io/qt-5/qpainter.html#setCompositionMode ) 中列出的所有 34 种合成模式。一些模式只是让 QColor 完全不透明,大多数与这个问题无关,它们都不允许我让两个矩形的交点保持与其他矩形相同的半透明颜色。

如果有人有想法,我将不胜感激。

您可以在下面找到重现问题的短代码:

from PyQt5.QtWidgets import QWidget, QApplication
from PyQt5.QtGui import QPainter, QColor
from PyQt5.QtCore import QRect
import sys


class Drawing(QWidget):
    def __init__(self):
        super().__init__()
        self.setGeometry(300, 300, 350, 300)
        self.show()

    def paintEvent(self, event):
            painter = QPainter(self)
            painter.setCompositionMode(QPainter.CompositionMode_SourceOver)
            painter.setPen(QColor(128, 128, 255, 128))
            painter.setBrush(QColor(128, 128, 255, 128))
            rect1 = QRect(10,10,100,100)
            rect2 = QRect(50,50,100,100)
            painter.drawRect(rect1)
            painter.drawRect(rect2)
                 
def main():
    app = QApplication(sys.argv)
    ex = Drawing()
    sys.exit(app.exec_())


if __name__ == '__main__':
    main() 

标签: pythonpyqtpyqt5qpainter

解决方案


简单的解决方案是使用 QPainterPath,确保它同时使用setFillRule(Qt.WindingFill)和使用simplified()合并所有子路径的路径版本:

path = QPainterPath()
path.setFillRule(Qt.WindingFill)
path.addRect(10, 10, 100, 100)
path.addRect(50, 50, 100, 100)
painter.drawPath(path.simplified())

推荐阅读