PySide/PyQt обрезает текст в QLabel на основе минимального размера

Мне интересно, как лучше всего обрезать текст в QLabel на основе его максимальной ширины/высоты. Входящий текст может быть любой длины, но для того, чтобы сохранить аккуратный макет, я хотел бы обрезать длинные строки, чтобы заполнить максимальное количество места (максимальная ширина/высота виджета).

E.g.:

 'A very long string where there should only be a short one, but I can't control input to the widget as it's a user given value'

станет:

'A very long string where there should only be a short one, but ...'

в зависимости от необходимого пространства для текущего шрифта.

Как я могу достичь этого лучше всего?

Вот простой пример того, что мне нужно, хотя это основано на количестве слов, а не на доступном месте:

import sys
from PySide.QtGui import *
from PySide.QtCore import *


def truncateText(text):
    maxWords = 10
    words = text.split(' ')
    return ' '.join(words[:maxWords]) + ' ...'

app = QApplication(sys.argv)

mainWindow = QWidget()
layout = QHBoxLayout()
mainWindow.setLayout(layout)

text = 'this is a very long string, '*10
label = QLabel(truncateText(text))
label.setWordWrap(True)
label.setFixedWidth(200)
layout.addWidget(label)

mainWindow.show()
sys.exit(app.exec_())

person Frank Rueter    schedule 12.07.2012    source источник


Ответы (3)


Еще проще — используйте метод QFontMetrics.elidedText и перегрузите paintEvent, вот пример:

from PyQt4.QtCore import Qt
from PyQt4.QtGui import QApplication,\
                        QLabel,\
                        QFontMetrics,\
                        QPainter

class MyLabel(QLabel):
    def paintEvent( self, event ):
        painter = QPainter(self)

        metrics = QFontMetrics(self.font())
        elided  = metrics.elidedText(self.text(), Qt.ElideRight, self.width())

        painter.drawText(self.rect(), self.alignment(), elided)

if ( __name__ == '__main__' ):
    app = None
    if ( not QApplication.instance() ):
        app = QApplication([])

    label = MyLabel()
    label.setText('This is a really, long and poorly formatted runon sentence used to illustrate a point')
    label.setWindowFlags(Qt.Dialog)
    label.show()

    if ( app ):
        app.exec_()
person Eric Hulser    schedule 01.08.2012
comment
@EricHulser, это очень хороший ответ. Очень полезно. Большое спасибо! - person Phil; 12.03.2013

Я обнаружил, что ответ @Eric Hulser, хотя и отличный, не работает, когда метка помещается в другой виджет.

Я придумал это, объединив ответ Эрика и Qt Пример ярлыка с пропуском. Как написано здесь, он позволяет передавать различные режимы пропуска и сохраняет текст по вертикали (конечно, он удаляется по горизонтали!).

Этап компоновки реализован в соответствии с к документам, мне непонятно, поэтому я не могу говорить об этом очень хорошо. По сути, он проверяет, не выходит ли текст метки за пределы ширины метки; если это так, он исключает текст.

Также неясно, что имеется в виду под допустимая строка. Удаление этих проверок приводит к сбою приложения. Я предполагаю, что линия действительна, когда она не выходит за пределы виджета.

Если вы хотите использовать PySide,

  • PyQt5 -> PySide2
  • pyqtSignal -> Сигнал

В любом случае, наслаждайтесь!

import sys
from PyQt5 import QtCore, QtWidgets, QtGui


class EliderLabel(QtWidgets.QLabel):

    elision_changed = QtCore.pyqtSignal(bool)

    def __init__(self, text='', mode=QtCore.Qt.ElideRight, **kwargs):
        super().__init__(**kwargs)

        self._mode = mode
        self.elided = False

        self.setSizePolicy(QtWidgets.QSizePolicy.Expanding, QtWidgets.QSizePolicy.Expanding)
        self.setText(text)

    def setText(self, text):
        self._contents = text
        # Changing the content require a repaint of the widget (or so
        # says the overview)
        self.update()

    def text(self):
        return self._contents

    def minimumSizeHint(self):
        metrics = QtGui.QFontMetrics(self.font())
        return QtCore.QSize(0, metrics.height())

    def paintEvent(self, event):

        super().paintEvent(event)

        did_elide = False

        painter = QtGui.QPainter(self)
        font_metrics = painter.fontMetrics()
        # fontMetrics.width() is deprecated; use horizontalAdvance
        text_width = font_metrics.horizontalAdvance(self.text())

        # Layout phase, per the docs
        text_layout = QtGui.QTextLayout(self._contents, painter.font())
        text_layout.beginLayout()

        while True:

            line = text_layout.createLine()

            if not line.isValid():
                break

            line.setLineWidth(self.width())

            if text_width >= self.width():
                elided_line = font_metrics.elidedText(self._contents, self._mode, self.width())
                painter.drawText(QtCore.QPoint(0, font_metrics.ascent()), elided_line)
                did_elide = line.isValid()
                break
            else:
                line.draw(painter, QtCore.QPoint(0, 0))

        text_layout.endLayout()

        self.elision_changed.emit(did_elide)

        if did_elide != self.elided:
            self.elided = did_elide
            self.elision_changed.emit(did_elide)


class MyDialog(QtWidgets.QWidget):

    def __init__(self):
        super().__init__()

        text = 'This is a really, long and poorly formatted runon sentence used to illustrate a point'
        label = EliderLabel(text, parent=self)

        label.elision_changed.connect(self.on_elide)

        layout = QtWidgets.QVBoxLayout()
        layout.addWidget(label)

        self.setLayout(layout)

    def on_elide(self, val):
        print('Elided: ', val, flush=True)


if __name__ == '__main__':
    app = QtWidgets.QApplication([])
    dia = MyDialog()
    dia.show()

    sys.exit(app.exec_())

person Lorem Ipsum    schedule 20.05.2021

Вы можете добиться этого, определив ширину с помощью QFontMetrics, см. этот ответ.

Вы, вероятно, захотите использовать или создать какой-нибудь алгоритм, который быстро находит место для вырезания, если только выполнение этого в простом цикле for не будет достаточным.

person Tim Meyer    schedule 12.07.2012