QT - 是否可以在QTableView中为一半的单元格着色?

时间:2017-07-27 18:29:34

标签: qt delegates qtableview

我想如果有可能它会涉及重新实现实际将颜色绘制到单元格中的函数,或者创建某种委托。

有人有这方面的经验吗?它甚至可能吗?

1 个答案:

答案 0 :(得分:3)

是的,完全可以按照您想要的方式绘制QTableView的单元格。 Qt足够灵活,允许这样做。

以下是一个如何通过自定义委托在PyQt5中完成的示例:

import sys
import string
import random
from PyQt5 import QtCore, QtWidgets, QtGui

class Delegate(QtWidgets.QStyledItemDelegate):
    def __init__(self):
        QtWidgets.QStyledItemDelegate.__init__(self)

    def paint(self, painter, option, index):
        painter.save()
        left_rect = QtCore.QRect(option.rect.left(), option.rect.top(),
                                 option.rect.width() / 2, option.rect.height())
        left_brush = QtGui.QBrush(QtCore.Qt.red)
        painter.fillRect(left_rect, left_brush)
        right_rect = QtCore.QRect(option.rect.left() + option.rect.width() / 2,
                                  option.rect.top(), option.rect.width() / 2,
                                  option.rect.height())
        right_brush = QtGui.QBrush(QtCore.Qt.blue)
        painter.fillRect(right_rect, right_brush)
        painter.restore()
        adjusted_option = option
        adjusted_option.backgroundBrush = QtGui.QBrush(QtCore.Qt.NoBrush)
        QtWidgets.QStyledItemDelegate.paint(self, painter, adjusted_option, index)

class Model(QtCore.QAbstractTableModel):
    def __init__(self):
        QtCore.QAbstractTableModel.__init__(self)
        self.column_names = [ "First",
                              "Second",
                              "Third",
                              "Fourth" ]
        self.text_data = []
        for i in range(0, 10):
            row_data = []
            for j in range(0, len(self.column_names)):
                row_data.append(''.join(random.choice(string.ascii_uppercase +
                                        string.digits) for _ in range(6)))
            self.text_data.append(row_data)

    def rowCount(self, parent):
        if parent.isValid():
            return 0
        else:
            return len(self.text_data)

    def columnCount(self, parent):
        return len(self.column_names)

    def data(self, index, role):
        if not index.isValid():
            return None
        if role != QtCore.Qt.DisplayRole:
            return None

        row = index.row()
        if row < 0 or row >= len(self.text_data):
            return None

        column = index.column()
        if column < 0 or column >= len(self.column_names):
            return None

        return self.text_data[row][column]

    def headerData(self, section, orientation, role):
        if role != QtCore.Qt.DisplayRole:
            return None
        if orientation != QtCore.Qt.Horizontal:
            return None
        if section < 0 or section >= len(self.column_names):
            return None
        else:
            return self.column_names[section]

class MainForm(QtWidgets.QMainWindow):
    def __init__(self, parent=None):
        QtWidgets.QMainWindow.__init__(self, parent)
        self.model = Model()
        self.delegate = Delegate()
        self.view = QtWidgets.QTableView()
        self.view.setModel(self.model)
        self.view.setItemDelegate(self.delegate)
        self.setCentralWidget(self.view)

def main():
    app = QtWidgets.QApplication(sys.argv)
    form = MainForm()
    form.show()
    app.exec_()

if __name__ == '__main__':
    main()

基本上,我们实现了一个自定义委托作为QStyledItemDelegate的子类,并重新实现paint方法来执行以下操作:

  1. 以红色
  2. 绘制单元格左半部分的背景
  3. 以蓝色
  4. 绘制单元格右半部分的背景
  5. 将绘画其余部分的背景画笔设置为“无画笔”,即防止进一步绘制背景
  6. 调用基类的绘画来绘制除了单元格背景之外的所有内容
  7. 这是最终结果:

    result

    更新。:这是翻译为C ++的相同代码的版本:Half-cell-delegate