Qt displayAlignment 没有右对齐

Qt displayAlignment not aligning to the right

我正在使用 PyQt 编写应用程序。我将 QListView 与 QStyledItemDelegate 一起使用,我需要在左侧对齐一些项目,在右侧对齐一些项目。默认情况下,所有这些都在左侧对齐。但是,我仍然不知道如何将它们右对齐。我发现这个问题 看起来像我需要的东西,但 displayAlignment 似乎并没有改变任何东西。这是我的代码:

import sys

from PyQt5.QtCore import (
    QAbstractListModel,
    QMargins,
    Qt,
    QSize,
)
from PyQt5.QtGui import (
    QPainter,
    QFont,
    QFontMetrics,
)
from PyQt5.QtWidgets import (
    QApplication,
    QListView,
    QMainWindow,
    QVBoxLayout,
    QWidget,
    QStyledItemDelegate,
)

BUBBLE_PADDING = QMargins(15, 5, 15, 5)
TEXT_PADDING = QMargins(25, 15, 25, 15)
FONT_SIZE = 14
FONT_STYLE = "Times"
font = 0

class MessageDelegate(QStyledItemDelegate):
    def initStyleOption(self, option, index):
        super(MessageDelegate, self).initStyleOption(option, index)
        option.displayAlignment = Qt.AlignRight

    def paint(self, painter, option, index):
        global font
        text = index.model().data(index, Qt.DisplayRole)
        option.rect.setSize(self.sizeHint(option, index))
        option.displayAlignment = Qt.AlignRight
        print(int(option.displayAlignment))
        bubblerect = option.rect.marginsRemoved(BUBBLE_PADDING)
        textrect = option.rect.marginsRemoved(TEXT_PADDING)
        painter.setPen(Qt.cyan)
        painter.setBrush(Qt.cyan)
        painter.drawRoundedRect(bubblerect, 10, 10)
        painter.setPen(Qt.black)
        painter.setFont(font)
        painter.drawText(textrect, Qt.TextWordWrap, text)

    def sizeHint(self, option, index):
        global font
        text = index.model().data(index, Qt.DisplayRole)
        metrics = QFontMetrics(font)
        rect = option.rect.marginsRemoved(TEXT_PADDING)
        rect = metrics.boundingRect(rect, Qt.TextWordWrap, text)
        rect = rect.marginsAdded(TEXT_PADDING)
        return rect.size()


class MessageModel(QAbstractListModel):
    def __init__(self, *args, **kwargs):
        super(MessageModel, self).__init__(*args, **kwargs)
        self.messages = []

    def data(self, index, role):
        if role == Qt.DisplayRole:
            return self.messages[index.row()]

    def rowCount(self, index):
        return len(self.messages)

    def add_message(self, text):
        if text:
            self.messages.append((text))
            self.layoutChanged.emit()


class MainWindow(QMainWindow):
    def __init__(self):
        global font
        super(MainWindow, self).__init__()
        font = QFont(FONT_STYLE, FONT_SIZE)
        self.resize(int(QApplication.primaryScreen().size().width() * 0.3), int(QApplication.primaryScreen().size().height() * 0.5))
        main_layout = QVBoxLayout()
        self.messages = QListView()
        self.messages.setItemDelegate(MessageDelegate())
        self.model = MessageModel()
        self.messages.setModel(self.model)
        self.model.add_message("Hello, world!")
        main_layout.addWidget(self.messages)
        self.w = QWidget()
        self.w.setLayout(main_layout)
        self.setCentralWidget(self.w)


app = QApplication(sys.argv)
window = MainWindow()
window.show()
app.exec_()

And here is the result. 如您所见,尽管在 paint 函数中打印的值为 2 (Qt.AlignRight),但消息仍然在左侧对齐。 (我分配了 Qt.AlignRight 2 次,因为函数 initStyleOption 从未被调用过)

设置样式选项的值只有在使用时才有意义,例如调用 QStyle 函数时。

您通过自己绘制所有内容来覆盖 paint,因此样式选项中的任何更改都是完全无用的。
相反,您应该更改除initStyleOption以外的任何函数中的选项(尤其是option.rect),而是创建选项(或其对象)的新实例) 基于作为参数收到的那个。

由于要将矩形右对齐,需要根据选项矩形的右减去所需的宽度,用其X坐标构造一个新矩形。

    def paint(self, painter, option, index):
        text = index.model().data(index, Qt.DisplayRole)
        size = self.sizeHint(option, index)
        <b>rect = QRect(
            option.rect.right() - size.width(), option.rect.y(), 
            size.width(), size.height())</b>
        bubblerect = rect.marginsRemoved(BUBBLE_PADDING)
        textrect = rect.marginsRemoved(TEXT_PADDING)
        painter.setPen(Qt.cyan)
        painter.setBrush(Qt.cyan)
        painter.drawRoundedRect(bubblerect, 10, 10)
        painter.setPen(Qt.black)
        painter.setFont(font)
        painter.drawText(textrect, Qt.TextWordWrap, text)

尽可能避免使用全局变量(如果你不知道如何使用它们,就不要使用它们,又名:如果你知道如何使用它们,你就不会使用它们他们)。如果要为委托设置字体,请使用首选字体为其 __init__ 添加参数,并将其设置为实例属性。