Я пытаюсь создать пользовательский класс TableModel
для QTableView
. Ячейки, содержащие 1
в качестве данных, должны иметь красный контур. Выделение выполняется путем возврата растрового изображения (с красными полями и текстом, нарисованным сверху) из TableModel
вместо возврата простой строки.
Проблема в неожиданном заполнении растрового изображения, которое я возвращаю как DecorationRole
. Я проверил, правильно ли отображается растровое изображение (и оно на самом деле имеет размер 21x21 пикселей с хорошо выполненным контуром, без отступов, как и планировалось) прямо перед линией return pixmap
.
Вот правильное нарисованное растровое изображение, которое было сохранено непосредственно перед return
из TableModel:
В конце концов, что-то сдвигает возвращаемое растровое изображение ровно на 3 пикселя от левой границы ячейки QTableView
. Я не установил никаких отступов в QtDesigner для QTableView
и не изменил их позже в своем коде. Я также попытался вручную установить нулевое заполнение, используя таблицу стилей, но это не дало другого результата.
Есть идеи как это исправить? Спасибо.
Вот образец моего TableModel
:
class TableModel(QtCore.QAbstractTableModel):
def __init__(self, topology=None):
super().__init__()
...
# Hardcode cell size and path to rectangle image
self.cell_width, self.cell_height = 21, 21
self.fpath_red_rect = './path/to/red_rect.png'
def rowCount(self, parent=QtCore.QModelIndex()):
return self.data.shape[0]
def columnCount(self, parent=QtCore.QModelIndex()):
return self.data.shape[1]
def headerData(self, section, orientation, role=QtCore.Qt.DisplayRole):
...
def size(self):
return QtCore.QSize((self.columnCount() + 1) * self.cell_width,
(self.rowCount() + 1) * self.cell_height)
def data(self, index, role=QtCore.Qt.DisplayRole):
if not index.isValid():
return QtCore.QVariant()
i = index.row()
j = index.column()
if role == QtCore.Qt.DisplayRole:
if self.data[i, j] == 0: # empty
return ''
elif self.data[i, j] == 1: # cell with red rectangle
# the text will be drawn on pixmap manually later
return None
else:
return '{0}'.format(self.data[i, j]) # display default data
if role == QtCore.Qt.DecorationRole:
# Create pixmap, draw the rectangle on it and then draw text on top
pixmap = QtGui.QPixmap(self.cell_width, self.cell_height)
image = QtGui.QImage(self.fpath_red_rect).scaled(self.cell_width, self.cell_height)
painter = QtGui.QPainter(pixmap)
painter.drawImage(pixmap.rect().topLeft(), image)
painter.drawText(pixmap.rect(), QtCore.Qt.AlignCenter, '{0}'.format(self.data[i, j]))
painter.end()
# If we save the pixmap to PNG image here (see the link above),
# we get the expected 21 x 21 px image, with nice
# and properly drawn rectangle and centered text.
# But something goes wrong after returning
return pixmap
if role == QtCore.Qt.BackgroundRole:
return QtGui.QBrush(self.getQtColor(self.data[i, j]))
if role == QtCore.Qt.TextAlignmentRole:
return QtCore.Qt.AlignCenter
return QtCore.QVariant()