【问题标题】:How to use QPainter in a QWidget in PyQt4如何在 PyQt4 的 QWidget 中使用 QPainter
【发布时间】:2017-04-29 06:03:59
【问题描述】:

我下面的代码当前打开了一个 500x500 的空白 QMainWindow。

我只是想用 QPainter 在 QWidget 中画一个圆。

这是我的代码:

from PyQt4 import QtCore, QtGui, Qt
from PyQt4.QtGui import QApplication, QMainWindow
import sys


class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.resize(500, 500)
        self.centralwidget = QtGui.QWidget(MainWindow)
        self.horizontalLayout = QtGui.QHBoxLayout(self.centralwidget)
        MainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtGui.QMenuBar(MainWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 500, 22))
        MainWindow.setMenuBar(self.menubar)
        self.statusbar = QtGui.QStatusBar(MainWindow)
        MainWindow.setStatusBar(self.statusbar)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)

class MyMainScreen(QMainWindow):
    def __init__(self, parent=None):
        QtGui.QMainWindow.__init__(self, parent)
        self.ui = Ui_MainWindow()  # This is from a python export from QtDesigner
        self.ui.setupUi(self)

        self.paintCircle()


    def paintCircle(self):
        self.painter = QtGui.QPainter(self)
        self.painter.begin(self)
        self.painter.setPen(Qt.QPen(Qt.QColor.black))
        # painter.end()
        self.painter.drawArc(QtCore.QRectF(250, 250, 10, 10), 0, 5760)


if __name__ == "__main__":
    app = QApplication(sys.argv)
    mainscreen = MyMainScreen()
    mainscreen.show()
    app.exec_()

我收到以下错误:

QPainter::begin: Paint device returned engine == 0, type: 1
QPainter::begin: Paint device returned engine == 0, type: 1
QPainter::setPen: Painter not active

【问题讨论】:

    标签: python pyqt pyqt4 qpainter


    【解决方案1】:

    在您的情况下,您必须覆盖 paintEvent 函数:

    from PyQt4 import QtCore, QtGui, Qt
    from PyQt4.QtGui import QApplication, QMainWindow
    import sys
    
    
    class Ui_MainWindow(object):
        def setupUi(self, MainWindow):
            MainWindow.resize(500, 500)
            self.centralwidget = QtGui.QWidget(MainWindow)
            self.horizontalLayout = QtGui.QHBoxLayout(self.centralwidget)
            MainWindow.setCentralWidget(self.centralwidget)
            self.menubar = QtGui.QMenuBar(MainWindow)
            self.menubar.setGeometry(QtCore.QRect(0, 0, 500, 22))
            MainWindow.setMenuBar(self.menubar)
            self.statusbar = QtGui.QStatusBar(MainWindow)
            MainWindow.setStatusBar(self.statusbar)
            QtCore.QMetaObject.connectSlotsByName(MainWindow)
    
    
    class MyMainScreen(QMainWindow):
        def __init__(self, parent=None):
            QtGui.QMainWindow.__init__(self, parent)
            self.ui = Ui_MainWindow()  # This is from a python export from QtDesigner
            self.ui.setupUi(self)
    
        def paintEvent(self, event):
            painter = QtGui.QPainter(self)
            painter.setPen(QtGui.QPen(QtCore.Qt.red))
            painter.drawArc(QtCore.QRectF(250, 250, 10, 10), 0, 5760)
    
    if __name__ == "__main__":
        app = QApplication(sys.argv)
        mainscreen = MyMainScreen()
        mainscreen.show()
        app.exec_()
    

    输出:

    【讨论】:

    • 如何让它在双击时触发?
    • 在框中,基本上我想我可以说如果 event == QtCore.QEvent.MouseButtonDblClick,然后 drawArc 但事件显然每次都是 QPaintEvent.paint
    • 我已经在使用需要 mouseDoubleClickEvent 的 eventFilter
    • 你能让 drawArc 在 matplotlib FigureCanvasQtAgg 上工作吗?当画布不存在时会显示圆圈,但是当我将以下三行添加到 init 时,它不再显示:
    • self.ui.centralwidget.fig = Figure((5,5), dpi=100) ########## self.ui.centralwidget.canvas = FigureCanvas(self. ui.centralwidget.fig) ######## self.ui.centralwidget.canvas.setParent(self.ui.centralwidget)
    【解决方案2】:

    你必须使用名字

    def paintEvent(self, event):
    

    而不是def paintCircle(self): 在需要时自动运行它。

    from PyQt4 import QtCore, QtGui, Qt
    from PyQt4.QtGui import QApplication, QMainWindow
    import sys
    
    class Ui_MainWindow(object):
    
        def setupUi(self, MainWindow):
            MainWindow.resize(500, 500)
            self.centralwidget = QtGui.QWidget(MainWindow)
            self.horizontalLayout = QtGui.QHBoxLayout(self.centralwidget)
            MainWindow.setCentralWidget(self.centralwidget)
            self.menubar = QtGui.QMenuBar(MainWindow)
            self.menubar.setGeometry(QtCore.QRect(0, 0, 500, 22))
            MainWindow.setMenuBar(self.menubar)
            self.statusbar = QtGui.QStatusBar(MainWindow)
            MainWindow.setStatusBar(self.statusbar)
            QtCore.QMetaObject.connectSlotsByName(MainWindow)
    
    class MyMainScreen(QMainWindow):
    
        def __init__(self, parent=None):
            QtGui.QMainWindow.__init__(self, parent)
            self.ui = Ui_MainWindow()  # This is from a python export from QtDesigner
            self.ui.setupUi(self)
    
        def paintEvent(self, event):
            self.ui.centralwidget.painter = QtGui.QPainter()
            self.ui.centralwidget.painter.begin(self)
            self.ui.centralwidget.painter.setPen(QtGui.QPen(QtCore.Qt.red))
            self.ui.centralwidget.painter.drawArc(QtCore.QRectF(250, 250, 10, 10), 0, 5760)
            self.ui.centralwidget.painter.end()
    
    if __name__ == "__main__":
        app = QApplication(sys.argv)
        mainscreen = MyMainScreen()
        mainscreen.show()
        app.exec_()
    

    示例draw on canvas

    【讨论】:

    • 没有必要像这样保留对QPainter 的引用:只需使用局部变量即可。
    • @ekhumoro 我使用了 OP 代码并做了最少的修改。
    猜你喜欢
    • 2017-04-12
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2018-02-04
    • 2016-09-23
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多