Python PyQt / PySide QMdiArea子窗口滚动在TabbedView中不起作用

时间:2013-08-23 15:49:59

标签: python-2.7 pyqt4 pyside qmdiarea

我使用PyQt设计器设置了简单示例。请参阅下文。 我有mdiarea in我在其中添加一个窗体作为子窗口。我使表单比主窗口长一点,看看子窗口是否显示滚动条。

问题: 如果我将mdiarea设置为setViewMode(QtGui.QMdiArea.TabbedView)滚动条停止工作并消失。 Howeevr如果我不使用TabbedView,滚动条工作正常。 谁能告诉我什么是错的?我需要带有工作滚动条的mdiarea的TabbedView。

我在win7上使用Python 2.7,PyQT 4.8.4 / PySide 1.2.1。

Python示例代码: 注释self.mdiArea.setViewMode行以查看示例工作。

import sys
from PyQt4 import QtCore, QtGui

class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName( "MainWindow" )
        MainWindow.resize(500, 400)
        self.centralwidget = QtGui.QWidget(MainWindow)
        self.centralwidget.setObjectName( "centralwidget" )
        self.verticalLayout = QtGui.QVBoxLayout(self.centralwidget)
        self.verticalLayout.setObjectName( "verticalLayout" )
        self.mdiArea = QtGui.QMdiArea(self.centralwidget)
        self.mdiArea.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded)
        self.mdiArea.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded)
        self.mdiArea.setActivationOrder(QtGui.QMdiArea.CreationOrder)
        self.mdiArea.setViewMode(QtGui.QMdiArea.TabbedView)
        self.mdiArea.setTabsClosable(True)
        self.mdiArea.setTabsMovable(True)
        self.mdiArea.setObjectName( "mdiArea" )
        self.verticalLayout.addWidget(self.mdiArea)
        MainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtGui.QMenuBar(MainWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 508, 21))
        self.menubar.setObjectName( "menubar" )
        self.menuAdd = QtGui.QMenu(self.menubar)
        self.menuAdd.setObjectName( "menuAdd" )
        MainWindow.setMenuBar(self.menubar)
        self.statusbar = QtGui.QStatusBar(MainWindow)
        self.statusbar.setObjectName( "statusbar" )
        MainWindow.setStatusBar(self.statusbar)
        self.menubar.addAction(self.menuAdd.menuAction())

        self.retranslateUi(MainWindow)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)

    def retranslateUi(self, MainWindow):
        MainWindow.setWindowTitle(   "MainWindow" )
        self.menuAdd.setTitle( "&Add Form" )

class Ui_Form(object):
    def setupUi(self, Form):
        Form.setObjectName( ("Form"))
        Form.resize(400, 800)
        self.gridLayout = QtGui.QGridLayout(Form)
        self.gridLayout.setObjectName( ("gridLayout"))
        self.plainTextEdit = QtGui.QPlainTextEdit(Form)
        self.plainTextEdit.setMinimumSize(QtCore.QSize(0, 731))
        self.plainTextEdit.setObjectName( ("plainTextEdit"))
        self.gridLayout.addWidget(self.plainTextEdit, 0, 0, 1, 1)
        self.buttonBox = QtGui.QDialogButtonBox(Form)
        self.buttonBox.setStandardButtons(QtGui.QDialogButtonBox.Cancel|QtGui.QDialogButtonBox.Ok)
        self.buttonBox.setObjectName( ("buttonBox"))
        self.gridLayout.addWidget(self.buttonBox, 1, 0, 1, 1)

        self.retranslateUi(Form)
        QtCore.QMetaObject.connectSlotsByName(Form)

    def retranslateUi(self, Form):
        Form.setWindowTitle( "Lengthy subwindow" )
        self.plainTextEdit.setPlainText( "Lengthy Form" ) 


class MyApp(QtGui.QMainWindow):
    def __init__(self, parent=None):
        super(MyApp, self).__init__(parent)
        self.ui = Ui_MainWindow()
        self.ui.setupUi(self)

    def Add_Subwindow(self):
        widget = QtGui.QWidget()
        self.subwin_abq = Ui_Form()
        self.subwin_abq.setupUi(widget)
        self.subwindow = QtGui.QMdiSubWindow(self.ui.mdiArea) 
        widget.setParent(self.subwindow)
        self.subwindow.setWidget(widget)  
        self.subwindow.setWindowTitle("testing")
        self.ui.mdiArea.addSubWindow(self.subwindow)
        widget.show()
        self.subwindow.show()
        self.subwindow.widget().show()

if __name__ == "__main__":
    app = QtGui.QApplication(sys.argv)
    window = MyApp()
    window.show()
    window.Add_Subwindow()
    sys.exit(app.exec_())

1 个答案:

答案 0 :(得分:0)

只是想感谢OP中的代码 - 在PyQT中寻找一个简单的MDI示例,而你的帮助我很多!我没有答案,但到目前为止我可以注意到这一点:我有Python 2.7,PyQT 4.8.3,只是评论setTabsClosablesetTabsMovable行,我可以得到你的例子如下所示:

qt1 qt2

我下载了designer-qt4并在那里查看了QMdiArea,似乎没有任何名为TabbedView的内容。所以我发现了这个:

QtWidgets 5.0: QMdiArea Class | Documentation | Qt Project

  

枚举ViewMode {SubWindowView,TabbedView}
  该枚举描述了该区域的视图模式;即如何显示子窗口   SubWindowView 0显示带窗框的子窗口(默认)。
  TabbedView 1在标签栏中显示带有标签的子窗口   documentMode:此属性保持标签栏是否在选项卡式视图模式下设置为文档模式。

我读到这个的方式:要么你要以MDI方式显示子窗口(所以它们可以比窗口大,带滚动条)子窗口变成制表符在选项卡式视图中 - 子窗口的大小不再重要,因此它会扩展为占用可用的选项卡区域。此外,在您的代码中,self.ui.mdiArea.documentMode()在两种情况下都会返回False

我还在MyApp.Add_Subwindow()

的末尾添加了此代码段
    sp = self.subwindow.sizePolicy()
    print sp.__dict__
    #print dir(sp)
    for attr in dir(sp):
      try:
        print "obj.%s = %s" % (attr, getattr(sp, attr))
      except: pass

这会转储一些有趣的数据(我不确定它们是否是对象属性):

obj.ButtonBox = 2
obj.CheckBox = 4
obj.ComboBox = 8
obj.ControlType = <class 'PyQt4.QtGui.ControlType'>
obj.ControlTypes = <class 'PyQt4.QtGui.ControlTypes'>
obj.DefaultType = 1
obj.ExpandFlag = 2
obj.Expanding = 7
obj.Fixed = 0
obj.Frame = 16
...

...但在运行选项卡式和MDI模式时,这些也不会改变。

那么,也许这是预期的行为?如果是这样,那就意味着你必须找到类似“孤独”标签显示小部件的东西;以编程方式添加几个QMdiAreas;隐藏所有这些,但在开始时隐藏默认值;然后绑定相应选项卡上的单击以显示“他们的”QMdiArea并隐藏其他人(但不用说,我还没有测试过它)。