Python PyQt/PySide QMdiArea прокрутка подокна не работает в TabbedView

У меня есть простой пример настройки с использованием PyQt designer. См. Ниже. У меня есть mdiarea, в котором я добавляю форму в качестве подокна. Я сделал форму немного длиннее основного окна, чтобы увидеть, появляется ли полоса прокрутки для дочернего подокна.

ПРОБЛЕМА: Если я установлю mdiarea на setViewMode(QtGui.QMdiArea.TabbedView) полосы прокрутки перестают работать и исчезают. Howeevr Если я не использую TabbedView, полосы прокрутки работают нормально. Может кто-нибудь сказать мне, что не так? Мне нужен TabbedView mdiarea с рабочими полосами прокрутки.

Я использую Python 2.7,PyQT 4.8.4/PySide 1.2.1 на win7.

Пример кода 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 ответ

Просто хотел сказать спасибо за код в OP - искал простой пример MDI в PyQT, и ваш мне очень помог! У меня нет точного ответа, но это то, что я могу отметить: у меня есть Python 2.7,PyQT 4.8.3, и я просто комментирую setTabsClosable а также setTabsMovable строка, я мог бы получить ваш пример, чтобы показать так:

QT1qt2

Я загрузил designer-qt4 и посмотрел там о QMdiArea кажется, ничего не называется TabbedView, Итак, я нашел это:

QtWidgets 5.0: QMdiArea Class | Документация | Qt Project

enum 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 и скрыть остальные (но, разумеется, я не проверял его).

Другие вопросы по тегам