Qt - 获取QLabel中字符串的像素长度

时间:2011-12-26 06:23:14

标签: python pyqt width pyqt4 qlabel

我有一个固定宽度的QLabel 我需要(定期)检查整个字符串是否以当前宽度放入QLabel内部,因此我可以适当调整它的大小。

为此,我需要获得字符串的'像素长度' (显示字符串所需的水平像素总数) 应该注意的是,QLabel的点大小永远不会改变。

Example of 'Pixel Width' of a string

我不能简单地检查存在的字符数量,因为有些字符是下标/上标,并且对整个字符串的宽度有不同的贡献。
(这就是说像素宽度和字符数量之间没有简单的关系)

这有什么抽象的,超级方便的功能吗?

规格:
Python 2.7.1
PyQt4中
Windows 7

1 个答案:

答案 0 :(得分:22)

要获得文本的精确像素宽度,您必须使用QFontMetrics.boundingRect

请勿使用QFontMetrics.width,因为它会考虑角色的左右方位。这通常会(但不总是)导致结果可能比整个像素宽度多或少几个像素。

因此,要计算标签文本的像素宽度,请使用以下内容:

width = label.fontMetrics().boundingRect(label.text()).width()

修改

有三种不同的QFontMetrics方法可用于计算字符串的“宽度”:size()width()boundingRect()

然而,尽管它们都给出了稍微不同的结果,但它们中没有一个似乎始终在所有情况下都返回确切的像素宽度。哪一个最好取决于当前使用的font-family以及字符串开头和结尾的特定字符。

我在下面添加了一个测试这三种方法的脚本。对我来说,boundingRect方法可以提供最一致的结果。其他两种方法往往略微过宽,或者在使用serif字体时剪切第二个文本样本(这是在Linux上使用PyQt 4.9和Qt 4.8)。

from PyQt4 import QtGui, QtCore

class Window(QtGui.QWidget):
    def __init__(self):
        QtGui.QWidget.__init__(self)
        self.setAutoFillBackground(True)
        self.setBackgroundRole(QtGui.QPalette.Mid)
        self.setLayout(QtGui.QFormLayout(self))
        self.fonts = QtGui.QFontComboBox(self)
        self.fonts.currentFontChanged.connect(self.handleFontChanged)
        self.layout().addRow('font:', self.fonts)
        for text in (
            u'H\u2082SO\u2084 + Be',
            u'jib leaf jib leaf',
            ):
            for label in ('boundingRect', 'width', 'size'):
                field = QtGui.QLabel(text, self)
                field.setStyleSheet('background-color: yellow')
                field.setAlignment(QtCore.Qt.AlignCenter)
                self.layout().addRow(label, field)
        self.handleFontChanged(self.font())

    def handleFontChanged(self, font):
        layout = self.layout()
        font.setPointSize(20)
        metrics = QtGui.QFontMetrics(font)
        for index in range(1, layout.rowCount()):
            field = layout.itemAt(index, QtGui.QFormLayout.FieldRole).widget()
            label = layout.itemAt(index, QtGui.QFormLayout.LabelRole).widget()
            method = label.text().split(' ')[0]
            text = field.text()
            if method == 'width':
                width = metrics.width(text)
            elif method == 'size':
                width = metrics.size(field.alignment(), text).width()
            else:
                width = metrics.boundingRect(text).width()
            field.setFixedWidth(width)
            field.setFont(font)
            label.setText('%s (%d):' % (method, width))

if __name__ == '__main__':

    import sys
    app = QtGui.QApplication(sys.argv)
    window = Window()
    window.show()
    sys.exit(app.exec_())