使用委托(PySide / Qt / PyQt)垂直居中项目的文本

时间:2015-10-02 14:22:03

标签: qt pyqt pyside

我有QTableView的自定义委托,允许显示/编辑html字符串(使用QTextDocument)。下面有一个SSCCE。

不幸的是,当我使用paint()显示文字时,它不是垂直居中的,而是似乎顶部对齐。例如,如果我将委托应用于第二列,而不是第一列,则表格如下所示:

the problem

我的搜索没有透露任何有关如何以委托方式从代理中解决此问题的方法。手动将5添加到option.rect.y()可以在我的计算机上运行,​​但我不认为这是有原则的。

有没有办法让文字垂直居中?

SSCCE

from PySide import QtGui
import sys

class HtmlTable(QtGui.QTableView):
    def __init__(self, parent = None):    
        QtGui.QTableView.__init__(self)
        model = QtGui.QStandardItemModel()
        model.setHorizontalHeaderLabels(['Title', 'Summary'])
        item0 = [QtGui.QStandardItem('Infinite Jest'), QtGui.QStandardItem('Hello, <i>Hal</i>')]
        item00 = [QtGui.QStandardItem('Hamlet'), QtGui.QStandardItem('Best served <b>cold</b>')]
        model.appendRow(item0)
        model.appendRow(item00)          
        self.setModel(model)
        self.setItemDelegate(HtmlPainter(self))

class HtmlPainter(QtGui.QStyledItemDelegate):
    def __init__(self, parent=None):
        QtGui.QStyledItemDelegate.__init__(self, parent)
    def paint(self, painter, option, index):
        if index.column() == 1: 
            text = index.model().data(index) #default role is display
            palette = QtGui.QApplication.palette()
            document = QtGui.QTextDocument()
            document.setDefaultFont(option.font)
            #Set text (color depends on whether selected)
            if option.state & QtGui.QStyle.State_Selected:  
                displayString = "<font color={0}>{1}</font>".format(palette.highlightedText().color().name(), text) 
                document.setHtml(displayString)
            else:
                document.setHtml(text)
            #Set background color
            bgColor = palette.highlight().color() if (option.state & QtGui.QStyle.State_Selected)\
                     else palette.base().color()
            painter.save()
            painter.fillRect(option.rect, bgColor)
            painter.translate(option.rect.x(), option.rect.y())  #If I add +5 it works
            document.drawContents(painter)
            painter.restore()
        else:
            QtGui.QStyledItemDelegate.paint(self, painter, option, index)          


def main():
    app = QtGui.QApplication(sys.argv)
    myTable = HtmlTable()
    myTable.show()
    sys.exit(app.exec_())

if __name__ == "__main__":
    main()

1 个答案:

答案 0 :(得分:3)

您应该使用QTextDocument::setTextWidth设置文档的宽度。这允许您确定文本高度并使用它来计算偏移量:

document.setTextWidth(option.rect.width())
offset_y = (option.rect.height() - document.size().height())/2
painter.translate(option.rect.x(), option.rect.y() + offset_y) 
document.drawContents(painter) 

设置文本宽度也是必要的,否则当列不够宽时,文本将不会被包装。

您可能希望重新实现sizeHint以根据文档大小计算首选宽度和高度。