QSplitter()无法平等处理QScrollArea()和QFrame()

3

1. 问题说明

我正在尝试使用Qt的QSplitter()部件。我在PyQt5中建立了一个非常简单的示例项目,其中一个QSplitter()封装了左侧的一个QScrollArea()和右侧的一个QFrame():

enter image description here

我已经给予QScrollArea()QFrame()相等的拉伸因子,但是QSplitter()并不对它们平等对待。 QScrollArea()始终占用更多的空间。 我不知道为什么。

 

最小完整可复现示例

只需将下面的代码复制粘贴到一个.py脚本中并运行它。 我在Windows 10机器上使用Python 3.7PyQt5

from PyQt5.QtWidgets import *
from PyQt5.QtGui import *
from PyQt5.QtCore import *
import sys

class Scroller(QScrollArea):
    '''
    The Scroller(), will be first widget in the Splitter().

    '''
    def __init__(self):
        super().__init__()
        self.setSizePolicy(QSizePolicy(QSizePolicy.Expanding, QSizePolicy.Expanding))
        self.setStyleSheet("""
            QScrollArea {
                background-color:#fce94f;
                border-color:#c4a000;
                padding: 0px 0px 0px 0px;
                margin: 0px 0px 0px 0px;
            }
        """)
        self.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOn)
        self.setHorizontalScrollBarPolicy(Qt.ScrollBarAsNeeded)

        # 1. FRAME
        self.__frm = QFrame()
        self.__frm.setStyleSheet("QFrame { background: #ff25292d; border: none; }")
        self.__frm.setMinimumHeight(100)

        # 2. LAYOUT
        self.__lyt = QVBoxLayout()
        self.__frm.setLayout(self.__lyt)

        # 3. SELF
        self.setWidget(self.__frm)
        self.setWidgetResizable(True)
        return

class Frame(QFrame):
    '''
    The Frame(), will be second widget in the Splitter()

    '''
    def __init__(self):
        super().__init__()
        self.setSizePolicy(QSizePolicy(QSizePolicy.Expanding, QSizePolicy.Expanding))
        # 1. FRAME
        self.setStyleSheet("""
            QFrame {
                background-color:#fcaf3e;
                border-color:#ce5c00;
                padding: 0px 0px 0px 0px;
                margin: 0px 0px 0px 0px;
            }
        """)
        self.__lyt = QVBoxLayout()
        self.__lyt.setAlignment(Qt.AlignTop)
        self.__lyt.setSpacing(0)
        self.__lyt.setContentsMargins(10, 10, 10, 10)
        self.setLayout(self.__lyt)
        return

class Splitter(QSplitter):
    '''
    The Splitter().

    '''
    def __init__(self, widg1, widg2):
        super().__init__()
        self.setOrientation(Qt.Horizontal)
        self.addWidget(widg1)
        self.addWidget(widg2)
        self.setStretchFactor(0, 5)
        self.setStretchFactor(1, 5)
        return

    def createHandle(self):
        return QSplitterHandle(self.orientation(), self)

class CustomMainWindow(QMainWindow):
    '''
    CustomMainWindow(), a QMainWindow() to start the whole setup.

    '''
    def __init__(self):
        super().__init__()
        self.setGeometry(100, 100, 600, 300)
        self.setWindowTitle("QSPLITTER TEST")

        # 1. OUTER FRAME
        self.__frm = QFrame()
        self.__frm.setSizePolicy(QSizePolicy.Expanding, QSizePolicy.Expanding)
        self.__frm.setStyleSheet("""
            QFrame {
                background-color: #eeeeec;
                border-color: #2e3436;
            }
        """)
        self.__lyt = QVBoxLayout()
        self.__frm.setLayout(self.__lyt)
        self.setCentralWidget(self.__frm)
        self.show()

        # 2. WIDGETS TO BE PUT IN SPLITTER
        self.__widg1 = Scroller()
        self.__widg2 = Frame()

        # 3. SPLITTER
        self.__splitter = Splitter(self.__widg1, self.__widg2)
        self.__lyt.addWidget(self.__splitter)
        return

if __name__== '__main__':
    app = QApplication(sys.argv)
    QApplication.setStyle(QStyleFactory.create('Plastique'))
    myGUI = CustomMainWindow()
    sys.exit(app.exec_())
1个回答

4

QSplitter不仅以拉伸因子为参考,还考虑了sizeHint()。如果添加以下内容:

# ...
# 3. SPLITTER
self.__splitter = Splitter(self.__widg1, self.__widg2)
self.__lyt.addWidget(self.__splitter)
print(self.__widg1.sizeHint(), self.__widg2.sizeHint())
return

你将得到以下内容:
PyQt5.QtCore.QSize(38, 22) PyQt5.QtCore.QSize(20, 20)

我们可以看到,在sizeHint()中,QScrollArea的宽度比QFrame大,这也解释了我们观察到的行为。
解决方案是要确保sizeHint()具有相同的宽度,与其所包含的内容无关。
class Scroller(QScrollArea):
    # ...

    def sizeHint(self):
        s = super().sizeHint()
        s.setWidth(20) # same width
        return s

class Frame(QFrame):
    # ...

    def sizeHint(self):
        s = super().sizeHint()
        s.setWidth(20) # same width
        return s
# ...

输出:

输入图像描述


(注意:这是一个示例文本,如需翻译具体内容,请提供相关的文本。)

太棒了,你又做到了!你是我的救星@eyllanesc。我要捐赠更多的咖啡来保持你的头脑清醒 :-) - K.Mulier

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接