0

我想显示图像并在每次鼠标左键单击时在当前鼠标位置放置一个标记。

然而,下面的代码完成了这项工作,它只有在("self.imglabel.setScaledContents(True)")被注释时才有效。任何原因?

我必须在不同分辨率的各种图像上完成这项工作,我阅读以保持适当的纵横比并适当地显示我们需要使用的图像setScaledContents(True)。但是为什么不允许这样做呢update() (PaintEvent)

import sys
from PyQt5.QtCore import Qt, QPoint
from PyQt5.QtWidgets import QMainWindow, QApplication, QLabel, QSizePolicy, QMessageBox
from PyQt5.QtGui import QPixmap, QPainter, QPen, QColor, QImage, QPalette

class Menu(QMainWindow):
    def __init__(self):
        super().__init__()
        self.central_widget = QWidget()               # define central widget
        self.setCentralWidget(self.central_widget)
        self.vbox = QVBoxLayout(self.central_widget)       
        self.vbox.addWidget(self.imgWidget())
        self.vbox.addWidget(QPushButton("test"))

    def imgWidget(self):
        self.imglabel = QLabel()
        self.imglabel.setScaledContents(True)
        self.image = QImage("calib.jpeg")
        self.imagepix = QPixmap.fromImage(self.image)
        self.imglabel.setPixmap(self.imagepix)
        self.imglabel.mousePressEvent = self.imgMousePress
        return self.imglabel

    def imgMousePress(self, e):
        painter = QPainter(self.imglabel.pixmap())
        pen = QPen()
        pen.setWidth(10)
        pen.setColor(QColor('red'))
        painter.setPen(pen)
        painter.drawPoint(e.x(), e.y())
        painter.end()
        self.imglabel.update()

if __name__ == '__main__':
    app = QApplication(sys.argv)
    mainMenu = Menu()
    mainMenu.show()
    sys.exit(app.exec_())
4

1 回答 1

0

为了避免对 QLabel 的每个paintEvent 进行不必要的计算,只要scaledContents属性为 True,就会缓存缩放的图像,并自动丢弃所有绘画。

为避免这种情况,您应该使用现有的创建 QPixmap 的新实例,然后再次设置新绘制的像素图。请注意,如果图像被缩放,小部件坐标将不会反映像素图上的实际位置,因此您需要使用转换来获取要绘制的实际点。

    def imgMousePress(self, e):
        pm = QPixmap(self.imglabel.pixmap())
        painter = QPainter(pm)
        pen = QPen()
        pen.setWidth(10)
        pen.setColor(QColor('red'))
        painter.setPen(pen)
        transform = QTransform().scale(
            pm.width() / self.imglabel.width(), 
            pm.height() / self.imglabel.height())
        painter.drawPoint(transform.map(e.pos()))
        painter.end()
        self.imglabel.setPixmap(pm)

考虑如果宽高比与源图像不同,所有“点”都会变成拉伸的矩形,但这只是外观问题:如果稍后保存像素图,它们将再次成为正方形,因为保存基于源像素图。
如果您想在显示时保持其方形形状,则需要跟踪点并覆盖 paintEvent 以在标签上手动绘制它们。

于 2020-03-11T19:28:23.753 回答