6

如果用户搜索某些内容,我的自定义委托QTableWidget会突出显示匹配项。不幸的是,矩形位置通常并不适合这种情况发生在某些字符或短语上,或者取决于匹配的数量或前导字符串的大小。我找不到导致这种情况的具体原因。这是一个例子:例子.

这是我的绘画程序(尝试解决问题的所有尝试和错误有点混乱):

void custom_delegate::paint(QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index) const{

    const QTableWidget* table_widget = qobject_cast<const QTableWidget*>(qstyleoption_cast<const QStyleOptionViewItemV3*>(&option)->widget);
    const int cell_width = table_widget->columnWidth(index.column());

    // basic table cell rectangle
    QRect rect_a = option.rect;

    // adjust rectangle to match text begin
    QStyle* style;
    if(table_widget != 0){
        style = table_widget->style();
    }else{
        style = QApplication::style();
    }
    const int text_horizontal_margin = style->pixelMetric(QStyle::PM_FocusFrameHMargin, 0, table_widget) + 1;
    QRect rect_b = rect_a.adjusted(text_horizontal_margin, 0, -text_horizontal_margin, 0);

    // adjust rectangle to match text height
    QFont cell_font = index.model()->data(index, Qt::FontRole).value<QFont>();
    cell_font.setPointSize(9);
    QFontMetrics fm(cell_font);
    const int height = fm.height();

    rect_b.setY(rect_a.y() + (rect_a.height() - height)/2);
    rect_b.setHeight(height);

    // displayed text
    std::string cell_text = qstrtostr(fm.elidedText(index.model()->data(index, Qt::DisplayRole).toString(),Qt::ElideRight,rect_a.width()));
    int found_pos = find_ci(cell_text, this->filter_string, 0);
    int old_pos = 0;
    int found_width = 0;
    QRect rect_c = rect_b;

    // find occurence of filter string in cell_text
    while(found_pos != std::string::npos){

        std::string front = cell_text.substr(0, found_pos);
        rect_c.setX(rect_b.x() + fm.tightBoundingRect(QString::fromStdString(front)).width());
        rect_c.setWidth(fm.width(QString::fromStdString(cell_text.substr(found_pos, this->filter_string.size()))));
        painter->fillRect(rect_c, Qt::yellow);
        old_pos = found_pos+1;
        found_pos = find_ci(cell_text, this->filter_string, old_pos);
    }
}

注意: filter_string是搜索的字符串,find_ci只是一个std::string::find包含不区分大小写的包装器,但在这里并不重要,因为这个测试用例是完全小写的,我std::string用于非 qt 的东西。

编辑:对于我尝试的宽度计算fm.tightBoundingRect().width()fm.boundingRect.width()结果fm.width()不同但从未正确。

我使用 Qt 5.2

4

1 回答 1

4

就我而言,我通过以下 hack 获得了预期的结果:

auto initialRect = fm.boundingRect(text);
auto improvedRect = fm.boundingRect(initialRect, 0, text);

目前尚不完全清楚为什么其他重载boundingRect返回正确的结果,但可能只是偶然的,因为正如文档所述:

这个函数返回的边界矩形比简单boundingRect()函数计算的要大一些。此函数使用最大左右字体方位角,这是多行文本正确对齐所必需的。此外,fontHeight()lineSpacing()用于计算高度,而不是单个字符的高度。

您提出的width方法也将返回更大的结果,但它似乎不正确,因为它应该只在您需要下一个单词的位置时使用:

[...]width()返回到应该绘制下一个字符串的距离。

此外,有时是否将结果传递painter.device()QFontMetrics构造函数也很重要。

于 2015-08-18T16:55:58.740 回答