3

ファイルからカスタム クラスを参照しようとすると、このエラーが発生し.uiます。私がしていることの何が問題なのですか?

"QFormBuilder was unable to create a custom widget of the class 'TimelinePane'; defaulting to base class 'QWidget'." 

QWidgetファイルで指定しているレイアウトでポップアップし.uiます。問題はカスタムクラスだけです。

カスタム クラスの説明を追加するために、.uiファイルを手動で変更しました (<customwidgets>セクション全体を追加しました)。これが、同じ Q がまだ見つからないため、新しい質問を開く必要がある理由です。ファイル内のクラスのパスを疑ってい.uiますが、試したオプションはどれも機能しませんでした (コメントアウトした部分を参照してください)。また、使用pythonはここでは問題ではないと思いますが、完全にはわかりません。まだ試しC++ていません。

from PySide import QtGui  
from PySide import QtCore
from PySide import QtUiTools

class MyWidget(QtGui.QMainWindow):
    def __init__(self, *args):  
       apply(QtGui.QMainWindow.__init__, (self,) + args)

       loader = QtUiTools.QUiLoader()
       file = QtCore.QFile('./src/prove_qt_ui_file/prove_main_widget.ui') 
       file.open(QtCore.QFile.ReadOnly)
       self.myWidget = loader.load(file, self)
       file.close()
       self.setCentralWidget(self.myWidget)

if __name__ == '__main__':  
   import sys  
   import os
   print("Running in " + os.getcwd() + " .\n")
   app = QtGui.QApplication(sys.argv)  
   win  = MyWidget()  
   win.show()
   app.exec_()

probe_main_widget.ui

<?xml version="1.0" encoding="UTF-8" ?>
<ui version="4.0">
 <class>MyWidget</class>
 <widget class="QWidget" name="MyWidget">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>687</width>
    <height>698</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>Runtime Monitor</string>
  </property>
  <layout class="QVBoxLayout">
   <property name="spacing">
    <number>0</number>
   </property>
   <property name="margin">
    <number>0</number>
   </property>
   <item>
    <widget class="QSplitter" name="splitter">
     <property name="orientation">
      <enum>Qt::Vertical</enum>
     </property>
     <property name="handleWidth">
      <number>9</number>
     </property>
     <widget class="QTreeWidget" name="warn_tree">
      <attribute name="headerVisible">
       <bool>false</bool>
      </attribute>
      <column>
       <property name="text">
        <string notr="true">1</string>
       </property>
      </column>
     </widget>
     <widget class="QTreeWidget" name="tree_all_devices">
      <attribute name="headerVisible">
       <bool>false</bool>
      </attribute>
      <column>
       <property name="text">
        <string notr="true">1</string>
       </property>
      </column>
     </widget>
     <widget class="TimelinePane" name="timeline_pane" native="true">
      <property name="minimumSize">
       <size>
        <width>0</width>
        <height>80</height>
       </size>
      </property>
      <property name="whatsThis">
       <string extracomment="Timeline"/>
      </property>
     </widget>
    </widget>
   </item>
  </layout>
 </widget>
 <customwidgets>
  <customwidget>
   <class>TimelinePane</class>
   <extends>QWidget</extends>
<!--   <header>timeline_pane</header> --> <!-- NG -->
<!--   <header>prove_qt_ui_file.timeline_pane</header> --> <!-- NG -->
   <header>src.prove_qt_ui_file.timeline_pane</header>  <!-- NG -->
  </customwidget>
 </customwidgets>
 <resources/>
 <connections/>
</ui>

タイムライン_ペイン.py

from PySide.QtGui import QWidget, QGraphicsScene, QGraphicsView, QColor, QHBoxLayout, QPushButton
class TimelinePane(QWidget):
    def __init__(self, parent):
        super(TimelinePane, self).__init__()
        print '\tTimelinePane init 1' # This doesn't print.

(環境) Ubuntu 12.04、python 2.7.3

4

1 に答える 1

6

理論的には、カスタム ウィジェットを機能させるには、次のように呼び出すだけで済みます。

loader.registerCustomWidget(TimelinePane)

loader.load() を呼び出す前に。あなたの場合、関連するインポートステートメントを追加する必要があります:

from timeline_pane import TimelinePane

ただし、これを Ubuntu 12.04 x86_64 上の PySide 1.1.2 でテストしたところ、segfault が発生しました。YMMVなので、テストしてください。

私はこの回避策を実装することになりました: http://www.mail-archive.com/pyside@qt-project.org/msg00306.html

つまり、QUiLoader の creteWidget() メソッドをオーバーライドし、ウィジェットの class_name が self.availableWidgets() にない場合は、追加した customWidgets インスタンス変数に従って自分でインスタンス化します。

ところで、ui ファイルを直接編集する代わりに、Qt Designer で右クリックの [プロモート] 機能を利用することもできます。

于 2013-02-14T14:57:53.927 に答える