3

我正在使用 PySide/PyQt,但这是一个通用的 Qt 问题。

有没有办法设置 QFormLayout 以便标签垂直居中而无需显式创建 QLabel 并将其垂直大小策略设置为先扩展?当第 2 列中的小部件高于我的标签时,我希望我的标签与小部件垂直居中,而不是与其顶部对齐...

这是一个演示该问题的示例脚本。我将标签涂成红色以更好地展示它们的尺寸行为。

from PySide import QtCore, QtGui

app = QtGui.QApplication([])

widget = QtGui.QWidget()
widget.setStyleSheet("QLabel { background-color : red}")

layout = QtGui.QFormLayout()
layout.setFieldGrowthPolicy(QtGui.QFormLayout.AllNonFixedFieldsGrow)
layout.setLabelAlignment(QtCore.Qt.AlignCenter)

editor1 = QtGui.QLineEdit()
editor1.setFixedSize(300, 100)

editor2 = QtGui.QLineEdit()
editor2.setFixedSize(300, 100)

layout.addRow('Input', editor1)
layout.addRow('Longer Named Input', editor2)

widget.setLayout(layout)

widget.show()

app.exec_()

结果如下:

带有左上对齐标签的 QFormLayout 示例

这是一个示例,通过显式创建 QLabel 并为它们提供扩展大小策略来演示所需的结果:

from PySide import QtCore, QtGui

app = QtGui.QApplication([])

widget = QtGui.QWidget()
widget.setStyleSheet("QLabel { background-color : red}")

layout = QtGui.QFormLayout()
layout.setFieldGrowthPolicy(QtGui.QFormLayout.AllNonFixedFieldsGrow)
layout.setLabelAlignment(QtCore.Qt.AlignCenter)

editor1 = QtGui.QLineEdit()
editor1.setFixedSize(300, 100)

editor2 = QtGui.QLineEdit()
editor2.setFixedSize(300, 100)

label1 = QtGui.QLabel('Input')
expand = QtGui.QSizePolicy.Expanding
label1.setSizePolicy(expand, expand)

label2 = QtGui.QLabel('Longer Named Input')
label2.setSizePolicy(expand, expand)

layout.addRow(label1, editor1)
layout.addRow(label2, editor2)

widget.setLayout(layout)

widget.show()

app.exec_()

这就是结果......

带有垂直扩展标签的 QFormLayout

我试过QFormLayout.setLabelAlignment()似乎没有帮助。文档甚至提到 setLabelAlignment 仅对标签进行水平对齐(即使这样似乎也不会居中,只是向左或向右)。

顺便说一句,这导致我也尝试将水平对齐设置为居中,但事实证明这更加困难,因为标签不会水平扩展以填充空间(小标签不会扩展以匹配最大的标签)。我可以获得水平居中标签的唯一方法是在显示小部件后显式找到最大标签的宽度,然后将所有其他标签设置为具有相同的宽度......

labels = [layout.itemAt(i*2).widget() for i in range(layout.rowCount())]
max_width = max(label.width() for label in labels)
for w in labels:
    w.setFixedWidth(max_width)
    w.setAlignment(QtCore.Qt.AlignCenter)

结果是:

在此处输入图像描述

我在 QFormLayout 级别是否缺少任何将标签居中的内容?我是否必须制作 QLabels 并设置为扩展或事后打开扩展(如下所示)?感谢您的任何想法!

expand = QtGui.QSizePolicy.Expanding
labels = [layout.itemAt(i*2).widget() for i in range(layout.rowCount())]
for w in labels:
    w.setSizePolicy(expand, expand)
4

2 回答 2

5

“[2] 标签只能左对齐(默认情况下)或右对齐” - 我不相信这是真的。

你的代码不适合我,所以我无法测试你的确切代码。但是,此方法适用于其他小部件:注意使用|分隔各种位置命令。

label2.setAlignment(PyQt5.QtCore.Qt.AlignLeft|PyQt5.QtCore.Qt.AlignVCenter)

我知道这已经有几年了,所以你可能想出了另一种方法,但这种方法对我来说效果很好。

于 2019-10-01T21:42:07.560 回答
1

我认为您的问题没有优雅的解决方案。来自QFormLayout的源代码

void QFormLayoutPrivate::arrangeWidgets(const QVector<QLayoutStruct>& layouts, QRect &rect)
{
    // [...]
    if (label) {
        int height = layouts.at(label->vLayoutIndex).size;
        if ((label->expandingDirections() & Qt::Vertical) == 0) {
            /*
                If the field on the right-hand side is tall,
                we want the label to be top-aligned, but not too
                much. So we introduce a 7 / 4 factor so that it
                gets some extra pixels at the top.
            */
            height = qMin(height,
                      qMin(label->sizeHint.height() * 7 / 4,
                           label->maxSize.height()));
            }

   [1]  QSize sz(qMin(label->layoutWidth, label->sizeHint.width()), height);
        int x = leftOffset + rect.x() + label->layoutPos;
   [2]  if (fixedAlignment(q->labelAlignment(), layoutDirection) & Qt::AlignRight)
   [ ]      x += label->layoutWidth - sz.width();
   [ ]  QPoint p(x, layouts.at(label->vLayoutIndex).pos);
        // ### expansion & sizepolicy stuff

        label->setGeometry(QStyle::visualRect(layoutDirection, rect, QRect(p, sz)));
    }
    // [...]
}

我们在这里看到了什么?

  • [1] 标签不水平拉伸
  • [2] 标签只能左对齐(默认)或右对齐

因此,您必须以某种方式手动同步标签的宽度(例如设置固定宽度)或放弃QFormLayoutQGridLayout改用。

于 2016-01-07T13:45:00.463 回答