QUiLoader在PySide中与自定义小部件崩溃

1 投票
1 回答
1726 浏览
提问于 2025-04-18 04:08

我正在尝试把一个PySide的PyQtGraph小部件嵌入到用QT Creator创建的GraphicsView窗口里。但是,当我导入UI文件并在QT Creator中使用“提升为”功能时,出现了一个分段错误。

这个错误之前也有人遇到过,已经有人提出了一种解决方法,就是重写QUiLoader的createWidget()方法:

http://www.mail-archive.com/pyside@qt-project.org/msg00306.html
还有
在.ui文件中调用自定义类失败

不过,当我尝试这个解决方法时,还是遇到了分段错误。

有没有人知道为什么这个解决方法在我的代码中似乎不管用?或者有没有其他方法可以在动态导入UI文件的同时嵌入PyQtGraph?

非常感谢,

Taco

一些示例代码:

#!/usr/bin/python
import os
import sys

from PySide.QtUiTools import QUiLoader


import pyqtgraph as pg

SCRIPT_DIRECTORY = os.path.dirname(os.path.abspath(__file__))

class UiLoader(QUiLoader):
    def __init__(self, baseinstance):
        QUiLoader.__init__(self, baseinstance)
        self.baseinstance = baseinstance

    def createWidget(self, className, parent = None, name = ""):
        if className in QUiLoader.availableWidgets(self):
            widget = QUiLoader.createWidget(self, className, parent, name)
        else:
            if hasattr(self.baseinstance, "customWidgets"):
                if className in self.baseinstance.customWidgets.keys():
                    widget = self.baseinstance.customWidgets[className](parent)
                else:
                    raise KeyError("Unknown widget '%s'" % className)
            else:
                raise AttributeError("Trying to load custom widget '%s', but base instance '%s' does not specify custom widgets." % (className, repr(self.baseinstance)))

        if self.baseinstance is not None:
            setattr(self.baseinstance, name, widget)

        return widget


def loadUi(uifile, baseinstance=None):
    loader = UiLoader(baseinstance)
    loader.registerCustomWidget(pg.PlotWidget)
    widget = loader.load(uifile)
    QMetaObject.connectSlotsByName(widget)
    return widget

class MainWindow(QMainWindow):
    def __init__(self, parent=None):
        QMainWindow.__init__(self, parent)
        self.ui = loadUi(os.path.join(SCRIPT_DIRECTORY, 'example1.ui'), self)
        self.ui.plotBtn.clicked.connect(self.PlotTest)

    def PlotTest(self):
        self.ui.plottest.plot(np.random.normal(size=50), clear=True)

def main():
    app = QApplication.instance()
    if app is None:
        app = QApplication(sys.argv)
    window = MainWindow()
    window.show()
    app.exec_()

if __name__ == '__main__':
    main()

还有UI文件:example1.ui

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
 <class>MainWindow</class>
 <widget class="QMainWindow" name="MainWindow">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>800</width>
    <height>600</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>MainWindow</string>
  </property>
  <widget class="PlotWidget" name="centralwidget">
   <widget class="QGraphicsView" name="graphicsView">
    <property name="geometry">
     <rect>
      <x>20</x>
      <y>30</y>
      <width>761</width>
      <height>471</height>
     </rect>
    </property>
   </widget>
   <widget class="QPushButton" name="plotBtn">
    <property name="geometry">
     <rect>
      <x>670</x>
      <y>510</y>
      <width>114</width>
      <height>32</height>
     </rect>
    </property>
    <property name="text">
     <string>Plot</string>
    </property>
   </widget>
  </widget>
  <widget class="QMenuBar" name="menubar">
   <property name="geometry">
    <rect>
     <x>0</x>
     <y>0</y>
     <width>800</width>
     <height>22</height>
    </rect>
   </property>
  </widget>
  <widget class="QStatusBar" name="statusbar"/>
 </widget>
 <customwidgets>
  <customwidget>
   <class>PlotWidget</class>
   <extends>QWidget</extends>
   <header>pyqtgraph</header>
   <container>1</container>
  </customwidget>
 </customwidgets>
 <resources/>
 <connections/>
</ui>

1 个回答

1

你可以通过使用 pg.Qt.loadUiType('uicfile.ui') 来解决 PySide 在使用 QtUiLoader 时遇到的问题。下面是一个基于你代码的示例:

from PySide.QtGui import *
import pyqtgraph as pg
import numpy as np

formClass, baseClass = pg.Qt.loadUiType('example1.ui')
class MainWindow(baseClass):
    def __init__(self, parent=None):
        baseClass.__init__(self, parent)
        self.ui = formClass()
        self.ui.setupUi(self)
        self.setCentralWidget(self.ui.centralwidget)
        self.ui.plotBtn.clicked.connect(self.PlotTest)

    def PlotTest(self):
        self.ui.centralwidget.plot(np.random.normal(size=50), clear=True)

def main():
    app = QApplication.instance()
    if app is None:
        app = QApplication(sys.argv)
    window = MainWindow()
    window.show()
    app.exec_()

if __name__ == '__main__':
    main()

注意事项:

  • 你的 .ui 文件似乎构建得不太正确——你把 'centralwidget' 提升成了一个 PlotWidget,但我觉得你应该是想把 GraphicsView 提升成这个。

  • 我在使用 loadUiType 时遇到了一些 Unicode 问题,所以我不得不使用最新的 pyqtgraph 开发版本。你可能也需要这样做。

撰写回答