QUiLoader在PySide中与自定义小部件崩溃
我正在尝试把一个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 开发版本。你可能也需要这样做。