PySide/PyQt съкращава текст в QLabel въз основа на minimumSize

Чудя се как най-добре да съкратя текст в 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 Пример за заличен етикет. Както е написано тук, той позволява преминаването на различни elide режими и запазва текста вертикално (разбира се, елидиран е хоризонтално!).

Фазата на оформлението, внедрена съгласно към документите, не ми е ясно, така че не мога да говоря с това много добре. По принцип той проверява дали текстът на етикета не надхвърля ширината на етикета; ако го направи, той премахва текста.

Също така не е ясно какво се има предвид под валиден ред. Премахването на тези проверки води до срив на приложението. Предполагам, че линията е валидна, когато не се простира отвъд джаджата.

Ако искате да използвате 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