gpt4 book ai didi

c++ - 自定义 QStyledItemDelegate 绘制多种颜色的文本

转载 作者:太空宇宙 更新时间:2023-11-04 12:37:09 25 4
gpt4 key购买 nike

我想在 QTableWidget 中显示两列,显示两个字符串之间的差异(之前由一些 Levenshtein 距离算法计算)。这些部分存储在每个 QTableWidgetItem 的数据中,作为 QStringList。第一部分必须显示为黑色,接下来的部分必须显示为红色,然后再次交替显示黑色、红色等。

为此,我实现了一个 QStyledItemDelegate,它带有一个自定义的 paint() 函数,该函数最终调用了一个 drawText() 方法:

void DifferencesDelegate::drawText(QPainter *painter,
const QStyleOptionViewItem &option,
const QModelIndex &index) const
{
painter->save();

const QPen defaultPen = painter->pen();

QStyleOptionViewItem opt = option;
initStyleOption(&opt, index);
opt.text.clear();

QStyle *style = opt.widget ? opt.widget->style() : QApplication::style();
style->drawControl(QStyle::CE_ItemViewItem, &opt, painter, opt.widget);

opt.rect.moveRight(opt.rect.right() + 3);

int color = 1;
for (const QString &part : index.data(Qt::UserRole).toStringList()) {
color++;
color = color % 2;
if (color) {
painter->setPen(Qt::red);
} else {
painter->setPen(defaultPen);
}

style->drawItemText(painter, opt.rect, opt.displayAlignment, opt.palette, true, part);
opt.rect.moveRight(opt.rect.right() + painter->fontMetrics().width(part));
}

painter->restore();
}

只要列的宽度足够,这就会导致正确的绘制:

Column's width sufficient

但是一旦列变小,我就会出现困惑的溢出:

Column too narrow

这肯定是由于 opt.rect 应用于显示的每个部分,而不是整个文本。

唯一的问题是我不知道如何解决这个问题;-)任何帮助将不胜感激!提前致谢!

最佳答案

出乎意料的是,我设法解决了它;-)

void DifferencesDelegate::drawText(QPainter *painter, const QStyleOptionViewItem &option,
const QModelIndex &index) const
{
if (! index.data(Qt::UserRole).isValid()) {
TableDelegate::drawText(painter, option, index);
return;
}

painter->save();

const QPen defaultPen = painter->pen();

QStyleOptionViewItem opt = option;
initStyleOption(&opt, index);
opt.text.clear();

QStyle *style = opt.widget ? opt.widget->style() : QApplication::style();
style->drawControl(QStyle::CE_ItemViewItem, &opt, painter, opt.widget);

constexpr int offset = 3;
const QString ellipsis = QStringLiteral("…");
const int ellipsisWidth = painter->fontMetrics().width(ellipsis);
const int rightBorder = opt.rect.left() + opt.rect.width() - offset;

QRect drawRect;
int color = 1;
int partWidth;
bool overflow = false;

opt.rect.moveRight(opt.rect.right() + offset);

const QStringList parts = index.data(Qt::UserRole).toStringList();
const int partsCount = parts.count();
for (int i = 0; i < partsCount; i++) {
color++;
color = color % 2;
if (color) {
painter->setPen(Qt::red);
} else {
painter->setPen(defaultPen);
}

partWidth = painter->fontMetrics().width(parts.at(i));

drawRect = opt.rect;
if (drawRect.left() + partWidth + (i == partsCount - 1 ? 0 : ellipsisWidth) > rightBorder) {
drawRect.setWidth(rightBorder - drawRect.left() - ellipsisWidth);
overflow = true;
}

style->drawItemText(painter, drawRect, opt.displayAlignment, opt.palette, true,
parts.at(i));

if (overflow) {
drawRect.setLeft(rightBorder - ellipsisWidth);
drawRect.setWidth(ellipsisWidth);
style->drawItemText(painter, drawRect, opt.displayAlignment, opt.palette, true,
ellipsis);
break;
}

opt.rect.moveRight(opt.rect.right() + partWidth);
}

painter->restore();
}

关于c++ - 自定义 QStyledItemDelegate 绘制多种颜色的文本,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55923137/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com