Как нарисовать фон всей строки в QStyledItemDelegate?
-
19-09-2019 - |
Вопрос
У меня есть QTableView, для которого я устанавливаю собственный QStyledItemDelegate.
В дополнение к пользовательскому рисованию элемента я хочу задать цвет фона строки для состояний выбора/наведения.Я хочу выглядеть примерно так, как на этом снимке экрана KGet:Фон строки KGet http://www.binaryelysium.com/images/kget_background.jpeg
Вот мой код:
void MyDelegate::paint( QPainter* painter, const QStyleOptionViewItem& opt, const QModelIndex& index ) const
{
QBrush backBrush;
QColor foreColor;
bool hover = false;
if ( opt.state & QStyle::State_MouseOver )
{
backBrush = opt.palette.color( QPalette::Highlight ).light( 115 );
foreColor = opt.palette.color( QPalette::HighlightedText );
hover = true;
}
QStyleOptionViewItemV4 option(opt);
initStyleOption(&option, index);
painter->save();
const QStyle *style = option.widget ? option.widget->style() : QApplication::style();
const QWidget* widget = option.widget;
if( hover )
{
option.backgroundBrush = backBrush;
}
painter->save();
style->drawPrimitive(QStyle::PE_PanelItemViewItem, &option, painter, widget);
painter->restore();
switch( index.column() )
{
case 0: // we want default behavior
style->drawControl(QStyle::CE_ItemViewItem, &option, painter, widget);
break;
case 1:
// some custom drawText
break;
case 2:
// draw a QStyleOptionProgressBar
break;
}
painter->restore();
}
В результате каждая отдельная ячейка получает фон при наведении курсора мыши только тогда, когда указатель мыши находится над ней, а не над всей строкой.Это сложно описать, поэтому вот скриншот:Результат приведенного выше кода http://www.binaryelysium.com/images/loader_bg.jpeg
На этом изображении указатель мыши находился над самой левой ячейкой, отсюда и выделенный фон.но я хочу, чтобы фон был нарисован по всей строке.
Как я могу этого добиться?
Редактировать:Еще немного подумав, я понял, что состояние QStyle::State_MouseOver передается только для фактической ячейки, над которой находится мышь, и когда метод рисования вызывается для других ячеек в строке, QStyle::State_MouseOver не устанавливается.
Таким образом, возникает вопрос, существует ли QStyle::State_MouseOver_Ряд государство (ответ:нет), так как мне этого добиться?
Решение
Вам нужно указать представлению обновлять свои ячейки, когда указатель мыши находится над строкой, поэтому я бы предложил отслеживать это в вашей модели.Затем в событии рисования вы можете запросить эти данные из индекса модели, используя пользовательскую роль данных.
Другие советы
void TrackDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
QStyleOptionViewItem viewOption(option);
if (viewOption.state & QStyle::State_HasFocus)
viewOption.state = viewOption.state ^ QStyle::State_HasFocus;
QImage image(m_RowBackGroundImagePath);
QPixmap pixmap(m_RowBackGroundImagePath);
qDebug()<<"forward"<<pixmap.width()<<pixmap.height();
pixmap.scaled(option.rect.width(),option.rect.height());
qDebug()<<"back"<<pixmap.width()<<pixmap.height();
qDebug()<<option.rect.width()<<option.rect.height();
QBrush brush(pixmap);
painter->save();
painter->fillRect(option.rect, brush/*QColor(238, 233, 233, 255)*/);
painter->restore();
viewOption.rect = QRect(option.rect.x(), option.rect.y(), option.rect.width(), option.rect.height());
//viewOption.palette.setColor(QPalette::Text, QColor(Qt::red));
//viewOption.palette.setBrush ( QPalette::ButtonText, brush1);
QItemDelegate::paint(painter, viewOption,index);
int progress = index.model()->data(index,Qt::DisplayRole).toInt();
QStyleOptionProgressBar progressBarOption;
progressBarOption.rect = QRect(option.rect.x(), option.rect.y()+(SETHEIGHT - PROGRESSBARHEIGHT)/2, option.rect.width(), /*option.rect.height()*/PROGRESSBARHEIGHT);
//qDebug()<<progressBarOption.rect.x()<<progressBarOption.rect.y()<<progressBarOption.rect.height()<<progressBarOption.rect.width();
//qDebug()<<option.rect.x()<<option.rect.y()<<option.rect.height()<<option.rect.width();
progressBarOption.state |= QStyle::State_Enabled;
progressBarOption.direction = QApplication::layoutDirection();
progressBarOption.fontMetrics = QApplication::fontMetrics();
progressBarOption.minimum = 0;
progressBarOption.maximum = 100;
progressBarOption.textAlignment = Qt::AlignCenter;
progressBarOption.textVisible = true;
progressBarOption.progress = progress < 0 ? 0 : progress;
progressBarOption.text = QString().sprintf("%d%%", progressBarOption.progress);
QApplication::style()->drawControl(QStyle::CE_ProgressBar, &progressBarOption, painter);
break;
}