【问题标题】:maintain grid layout widget as square将网格布局小部件保持为正方形
【发布时间】:2021-05-22 18:05:54
【问题描述】:

这应该很容易,但我只是越来越困惑,看着建议的答案不起作用。我想要做的就是能够在网格布局中拥有一堆小部件,其中调整窗口大小将导致网格内的小部件被调整大小以尽可能填充空间但保留它们的纵横比(在这种情况下为正方形)这是一个 2x2 的正方形网格)。

这是没有任何失败尝试的代码。我最初计划将窗口本身限制为固定的纵横比,但也无法让它工作。我将满足于具有此功能的框架(或其他小部件/布局)。限制正方形大小并不重要,只要它随着窗口拉伸和收缩,同时保留正方形内容。我在这里使用 PySide6,但是使用 PyQt5 的解决方案就可以了。 C ++,虽然不是很多,但我懂它。是否需要自定义处理大小调整?我似乎无法让 heightForWidth 工作。

import sys
from PySide6.QtWidgets import QApplication, QDialog, QWidget, QFrame, QGridLayout


class MyDialog(QDialog):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.resize(600, 600)
        self.my_frame = QFrame(self)
        self.my_frame.setObjectName(u"my_frame")
        self.my_frame.setGeometry(50, 50, 250, 250)
        self.my_frame.setBaseSize(100, 100)
        self.gridLayout = QGridLayout(self.my_frame)
        self.gridLayout.setSpacing(0)
        self.gridLayout.setObjectName("gridLayout")

        self.top_left = QWidget(self.my_frame)
        self.top_left.setObjectName("top_left")
        self.top_left.setStyleSheet("background-color: rgb(235, 10, 30)")
        self.gridLayout.addWidget(self.top_left, 0, 0, 1, 1)

        self.top_right = QWidget(self.my_frame)
        self.top_right.setObjectName("top_right")
        self.top_right.setStyleSheet("background-color: rgb(55, 122, 70)")
        self.gridLayout.addWidget(self.top_right, 0, 1, 1, 1)

        self.bottom_left = QWidget(self.my_frame)
        self.bottom_left.setObjectName("bottom_left")
        self.bottom_left.setStyleSheet("background-color: rgb(55, 122, 190)")
        self.gridLayout.addWidget(self.bottom_left, 1, 0, 1, 1)

        self.bottom_right = QWidget(self.my_frame)
        self.bottom_right.setObjectName("bottom_right")
        self.bottom_right.setStyleSheet("background-color: rgb(235, 219, 70)")
        self.gridLayout.addWidget(self.bottom_right, 1, 1, 1, 1)


if __name__ == "__main__":
    app = QApplication(sys.argv)
    dialog = MyDialog()
    dialog.show()
    exit(app.exec_())

【问题讨论】:

  • PyQt5和PySide6不一样,首先PyQt5是Qt5的绑定,PySide6是Qt6的,Qt5和Qt6是不同的(Qt6相比Qt5有很多bug和默认行为变化)。此外,这两种绑定都使用不同的工具来构建:sip 和 shiboken。所以请不要同时使用这两个标签。
  • @eyllanesc 请停止从问题中删除对 PyQt 的引用。通过将其限制为 PySide 或 PyQt 来限制潜在答案是没有意义的。我对任何一个答案都很满意。我可以使用 PyQt5 导入来轻松提出问题。
  • 非常感谢您对我的问题的回答,但我不明白您在这里的态度。您说“在许多情况下,解决方案可能适用于两个图书馆,但不适用于其他图书馆”——这意味着不同的解决方案将帮助社区的不同成员,因此在一种环境中工作的人可能会选择切换以克服与另一种环境的问题。信息多于少。
  • 嗯,我认为我们看待事物的方式不同。对我来说,标签用于保持事物井井有条,所以如果我使用 X 技术,那么我应该使用 X 标签,否则我们将使用更通用的标签,如“编程”、“算法”等。标签出现在问题中所以它匹配是相关的。另一方面,如果它非常通用,它会使搜索变得困难,我们最终会得到数千个帖子,其中 OP 提出相同的问题,而社区的回答相同。
  • 您是从提问者的角度来看它,而且很多时候它不会寻找重复项,但我是从回答者的角度来看的。我不想从其他公众那里发布相同的答案,我更愿意将其标记为重复,因为这样除了更简单的搜索之外,我们还可以节省社区获取答案的时间。

标签: python qgridlayout pyside6


【解决方案1】:

QLayouts(如 QGridLayout)是大小处理程序,不是可视元素,因此说它们是方形的毫无意义。另一方面,QGridLayout 主要根据容器的大小来建立小部件的几何形状。

因此在这种情况下,您必须根据窗口大小使其具有方形外观,因此您必须重写 resizeEvent 方法并计算窗口中内接的最大正方形的大小。也许您还应该将容器居中。

class MyDialog(QDialog):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.resize(600, 600)
        qss = """
            #top_left{
                background-color: rgb(235, 10, 30)
            }
            #top_right{
                background-color: rgb(55, 122, 70)
            }
            #bottom_left{
                background-color: rgb(55, 122, 190)
            }
            #bottom_right{
                background-color: rgb(235, 219, 70)
            }
        """
        self.setStyleSheet(qss)

        self.my_frame = QFrame(self, objectName="my_frame")
        self.gridLayout = QGridLayout(self.my_frame, objectName="gridLayout")
        self.gridLayout.setSpacing(0)

        self.top_left = QWidget(objectName="top_left")
        self.gridLayout.addWidget(self.top_left, 0, 0, 1, 1)

        self.top_right = QWidget(objectName="top_right")
        self.gridLayout.addWidget(self.top_right, 0, 1, 1, 1)

        self.bottom_left = QWidget(objectName="bottom_left")
        self.gridLayout.addWidget(self.bottom_left, 1, 0, 1, 1)

        self.bottom_right = QWidget(objectName="bottom_right")
        self.gridLayout.addWidget(self.bottom_right, 1, 1, 1, 1)

    def resizeEvent(self, event):
        super().resizeEvent(event)

        l = min(self.width(), self.height())
        center = self.rect().center()

        rect = QRect(0, 0, l, l)
        rect.moveCenter(center)
        self.my_frame.setGeometry(rect)

【讨论】:

  • 谢谢。这就是诀窍。虽然需要“从 PySide6.QtCore 导入 QRect”这一行。是否有任何理由这不适用于 PyQt5?
  • @EugeneGill 请阅读我的其他评论,问题不在于该解决方案不适用于 PyQt,而是出于订单原因。如果帖子未分类,则倾向于发布相同的问题和答案,并且 SO 的目标是收集高质量的问答,任何用户都可以找到答案,如果信息混乱,则搜索会更复杂再次这样做会发布相同的问题,使社区更加不必要地努力
  • 在尝试找到不覆盖调整大小的解决方案时,我遇到了 heightForWidth 作为类似问题的部分解决方案。这似乎是一个红鲱鱼。
猜你喜欢
  • 2019-07-22
  • 2015-03-06
  • 2018-03-30
  • 1970-01-01
  • 2017-03-22
  • 1970-01-01
  • 1970-01-01
  • 2021-01-17
  • 1970-01-01
相关资源
最近更新 更多