当我最小化QMdiSubWindow时,一个FigureCanvasQTAgg嵌套在QMdiSubWindow段错误中

时间:2016-03-13 23:41:22

标签: python qt matplotlib pyqt qmdiarea

我正在尝试在QMdiSubWindow中使用FigureCanvasQTAgg,以便用户可以动态创建他/她自己的绘图。 我已经制作了这个非常小的自包含代码:

from PyQt4 import QtGui
from matplotlib.backends.backend_qt4agg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.figure import Figure
import sys

class ExampleApp(QtGui.QMainWindow):
    def __init__(self):
        super(self.__class__, self).__init__()
        self.mdiarea = QtGui.QMdiArea()
        self.setCentralWidget(self.mdiarea)
        sub = QtGui.QMdiSubWindow(self.mdiarea)
        fig = Figure()
        p = FigureCanvas(fig)
        sub.layout().addWidget(p)
        sub.show()

def main():
    app = QtGui.QApplication(sys.argv)
    form = ExampleApp()
    form.show()
    app.exec_()

if __name__ == '__main__':
    main()

当我运行程序并尝试最小化QtGui.QMdiSubWindow对象时,会出现问题。当我这样做时,程序段错误并退出,没有错误描述。 这可能是qt,python绑定或FigureCanvasQTAgg对象中的错误。当然也可能是我错误地使用这些对象。请帮助我理解为什么在我最小化子窗口时发生段错误并帮助我弄清楚如何解决这个问题。 谢谢。

我的环境是ubuntu 14.04并正在使用 Qt版本:4.8.7 SIP版本:4.16.9 PyQt版本:4.11.4 MatplotLib版本:1.5.0

以下是拖放属性集的示例。似乎也存在问题。

from PyQt4 import QtGui
from matplotlib.backends.backend_qt4agg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.figure import Figure
import sys

class QtZListView(QtGui.QListView):
    def __init__(self, *args, **kwargs):
        QtGui.QListView.__init__(self, *args, **kwargs)
        self.model = QtGui.QStringListModel(['a','b','c'])
        self.setEditTriggers(QtGui.QAbstractItemView.NoEditTriggers)
        self.setModel(self.model)
        self.setSelectionMode(QtGui.QAbstractItemView.ExtendedSelection)
        self.setDragEnabled(True)

    def setStringList(self, *args, **kwargs):
        return self.model.setStringList(*args, **kwargs)

class mplsubwindow(QtGui.QMdiSubWindow):

    def __init__(self, *args, **kwargs):
        QtGui.QMdiSubWindow.__init__(self, *args, **kwargs)
        self.setWindowTitle("testing")
        self.setAcceptDrops(True)
        self.resize(400,400)
        self.show()

    def dragEnterEvent(self, event):
        print('entering')
        super(mplsubwindow, self).dragEnterEvent(event)

    def dragMoveEvent(self, event):
        print('drag moving')
        super(mplsubwindow, self).dragMoveEvent(event)

    def dropEvent(self, event):
        print('dropped')
        super(mplsubwindow, self).dropEvent(event)

class ExampleApp(QtGui.QMainWindow):
    def __init__(self):
        super(self.__class__, self).__init__()
        mainwid = QtGui.QWidget()
        self.mdiarea = QtGui.QMdiArea()

        layout = QtGui.QGridLayout(mainwid)
        layout.addWidget(self.mdiarea)
        sub = mplsubwindow(self.mdiarea)
        sub.show()
        layout.addWidget(QtZListView())
        self.setCentralWidget(mainwid)
        #self.setWidget(mainwid)

def main():
    app = QtGui.QApplication(sys.argv)
    form = ExampleApp()
    form.show()
    app.exec_()

if __name__ == '__main__':
    main()

2 个答案:

答案 0 :(得分:1)

问题似乎是当最小化时,小部件具有负高度(我想这是有道理的,但我找不到任何关于这个事实的文档;我通过添加一些打印语句注意到这一点)。解决方案是在这些情况下不要画画。我已经提交了PR来修复此上游,但您可能需要使用monky patch matplotlib.backends.backend_qt5agg.FigureCanvasQTAggBase.__draw_idle_agg

def __draw_idle_agg(self, *args):
    if self.height() < 0 or self.width() < 0:
        self._agg_draw_pending = False
        return
    try:
        FigureCanvasAgg.draw(self)
        self.update()
    finally:
        self._agg_draw_pending = False

请注意,模块中的qt5不是拼写错误,Qt4功能来自Qt5支持。

答案 1 :(得分:1)

问题似乎围绕为matplotlib小部件报告的大小不正确。正如@tcaswell指出的那样,matplotlib应该被修复以确保它不会导致段错误。

我将从另一方面解决问题并尝试阻止Qt报告虚假维度。似乎使用“内置”布局会导致问题。这可能是因为布局的存在是由QMdiSubWindow的{​​{1}}继承的,但QWidget的实现可能无法正确使用它。只要您使用QMdiSubWindow方法并创建自己的布局,就可以避免使用段错误。

以下是一些示例代码,其中包含您自己管理的布局:

QMdiSubWindow.setWidget()

修改

如果查看underlying C++ implementation,您可以看到对p = FigureCanvas(fig) container = QtGui.QWidget() layout = QtGui.QVBoxLayout(container) layout.addWidget(p) sub.setWidget(container) 的调用比将小部件添加到布局要复杂得多!