基于pyqt样式表的TableView单元格基于内容的样式

时间:2018-10-11 11:51:50

标签: python pyqt pyqt5

我发现了一种基于单元格内容在TableView中基于CSS的单元格样式的方法。以下代码显示了一个示例:

#!/usr/bin/python3

from PyQt5 import QtWidgets, QtGui, QtCore

class_values = ["zero", "one", "two"]

class Cell(QtWidgets.QWidget):
    def initFromItem(self, item):
        self.setProperty('dataClass', class_values[int(item.text())])

class TDelegate(QtWidgets.QStyledItemDelegate):
    def __init__(self, *a):
        super(TDelegate, self).__init__(*a)
        self.cell = Cell(self.parent())

    def paint(self, painter, option, index):
        item = index.model().itemFromIndex(index)
        self.cell.initFromItem(item)
        self.initStyleOption(option, index)
        style = option.widget.style() if option.widget else QtWidgets.QApplication.style()
        style.unpolish(self.cell)
        style.polish(self.cell)
        style.drawControl(QtWidgets.QStyle.CE_ItemViewItem, option, painter, self.cell)

class TTableModel(QtGui.QStandardItemModel):
    def __init__(self, parent=None):
        super(TTableModel, self).__init__(parent)
        for i in range(5):
            self.appendRow([QtGui.QStandardItem(str((x+i) % 3)) for x in range(5)])

class TTableView(QtWidgets.QTableView):
    def __init__(self, parent=None):
        super(TTableView, self).__init__(parent)
        self.setItemDelegate(TDelegate(self))

class Main(QtWidgets.QMainWindow):
    def __init__(self):
        super(Main, self).__init__()
        self.table = TTableView(self)
        self.model = TTableModel(self)
        self.table.setModel(self.model)
        self.setCentralWidget(self.table)

if __name__ == "__main__":
    import sys

    app = QtWidgets.QApplication(sys.argv)
    app.setStyleSheet("""
Cell[dataClass=zero]::item { background-color: gray; }
Cell[dataClass=one]::item { background-color: green; font-style: italic }
Cell[dataClass=two]::item { font-weight: bold }
""")
    mainWin = Main()
    mainWin.show()
    sys.exit(app.exec_())

这会生成一个像这样的表:

具有每个单元格样式的TableView

问题在于,尽管颜色有效,但字体样式无效。我究竟做错了什么?如何改善我的代码?以及它如何运作?例如,为什么CSS选择器必须包含::item。感谢所有收到的答案。但是请记住,对基于CSS的样式的需求对于项目至关重要。

1 个答案:

答案 0 :(得分:1)

这是由于qt(v5.9.5)中的错误导致在创建CE_ItemViewItem时忽略了所有字体样式信息(请参见QStyleSheetStyle::drawControl)。通过创建诸如CE_ToolBoxTabLabel之类的东西来作弊(它可以正确处理drawControl中的字体)可以使您获得字体格式,但可以使您上色,因为渲染使用的是按钮面板,而不是按钮面板。在选项(或关联的CSS)中指定。因此,您可以拥有一个,但不能同时拥有两个。我知道没有解决方法。

这是如何工作的。在QStyleSheetStyle::drawControl的{​​{1}}中,查找CE_ItemViewItem子角色的CSS,如果存在,将其应用于选项的副本(但不包括字体样式),然后应用于Item根据更新的选项及其更新的调色板绘制。不幸的是,由于无法从PyQt应用样式表,因此无法破解此代码(因为QStyleSheet不属于Qt的公共API)。