2021-05-25 19:42:40 +08:00
|
|
|
|
#include "result-view-delegate.h"
|
2022-02-24 19:58:47 +08:00
|
|
|
|
#include <QPainterPath>
|
2021-12-14 14:43:35 +08:00
|
|
|
|
using namespace UkuiSearch;
|
2021-12-13 10:53:28 +08:00
|
|
|
|
static ResultItemStyle *global_instance_of_item_style = nullptr;
|
2021-05-25 19:42:40 +08:00
|
|
|
|
|
|
|
|
|
ResultViewDelegate::ResultViewDelegate(QObject *parent) : QStyledItemDelegate(parent)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ResultViewDelegate::setSearchKeyword(const QString ®FindKeyWords)
|
|
|
|
|
{
|
|
|
|
|
m_regFindKeyWords.clear();
|
|
|
|
|
m_regFindKeyWords = regFindKeyWords;
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-05 14:27:14 +08:00
|
|
|
|
QSize ResultViewDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const
|
|
|
|
|
{
|
|
|
|
|
QSize size = QStyledItemDelegate::sizeHint(option,index);
|
|
|
|
|
size.setHeight(size.height() + 10);
|
|
|
|
|
return size;
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-25 19:42:40 +08:00
|
|
|
|
void ResultViewDelegate::paint(QPainter * painter, const QStyleOptionViewItem & option, const QModelIndex & index) const {
|
2022-05-19 13:47:54 +08:00
|
|
|
|
QStyleOptionViewItem opt = option;
|
|
|
|
|
initStyleOption(&opt, index);
|
|
|
|
|
|
|
|
|
|
QStyle *style = opt.widget ? opt.widget->style() : QApplication::style();
|
|
|
|
|
|
|
|
|
|
QString text = opt.text;
|
|
|
|
|
if(text.isEmpty()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
opt.text = QString();
|
|
|
|
|
style->drawControl(QStyle::CE_ItemViewItem, &opt, painter); //绘制非文本区域内容
|
|
|
|
|
// style->drawPrimitive(QStyle::PE_PanelItemViewItem, qstyleoption_cast<QStyleOption *>(&opt), painter, opt.widget);
|
|
|
|
|
|
|
|
|
|
opt.text = text;
|
|
|
|
|
QTextDocument doc;
|
|
|
|
|
doc.setHtml(getHtmlText(painter, opt, index)); //提取富文本
|
|
|
|
|
QAbstractTextDocumentLayout* layout = doc.documentLayout();
|
|
|
|
|
const double height = layout->documentSize().height();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
QRect textRect = style->subElementRect(QStyle::SE_ItemViewItemText, &opt, opt.widget);
|
|
|
|
|
//使图标和文本间隔与原来保持一致,故文本区域右移4
|
|
|
|
|
textRect.adjust(4, 0, 0, 0);
|
|
|
|
|
double y = textRect.y();
|
|
|
|
|
y += (textRect.height() - height) / 2;
|
|
|
|
|
|
|
|
|
|
QAbstractTextDocumentLayout::PaintContext context;
|
|
|
|
|
|
|
|
|
|
QPalette::ColorGroup cg = opt.state & QStyle::State_Enabled
|
|
|
|
|
? QPalette::Normal : QPalette::Disabled;
|
|
|
|
|
if (cg == QPalette::Normal && !(opt.state & QStyle::State_Active))
|
|
|
|
|
cg = QPalette::Inactive;
|
|
|
|
|
|
|
|
|
|
if(opt.state & QStyle::State_Selected) {
|
|
|
|
|
painter->setPen(opt.palette.color(cg, QPalette::HighlightedText));
|
|
|
|
|
} else {
|
|
|
|
|
painter->setPen(opt.palette.color(cg, QPalette::Text));
|
|
|
|
|
}
|
|
|
|
|
painter->save();
|
|
|
|
|
painter->translate(QPointF(textRect.x(), y));
|
|
|
|
|
layout->draw(painter, context); //绘制文本区域内容
|
|
|
|
|
painter->restore();
|
|
|
|
|
|
2021-05-25 19:42:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString ResultViewDelegate::getHtmlText(QPainter *painter, const QStyleOptionViewItem &itemOption, const QModelIndex &index) const
|
|
|
|
|
{
|
|
|
|
|
int indexFindLeft = 0;
|
|
|
|
|
QString indexString = index.model()->data(index, Qt::DisplayRole).toString();
|
|
|
|
|
QFont ft(painter->font().family(), GlobalSettings::getInstance()->getValue(FONT_SIZE_KEY).toInt());
|
|
|
|
|
QFontMetrics fm(ft);
|
2021-12-17 17:39:37 +08:00
|
|
|
|
QString indexColString = fm.elidedText(indexString, Qt::ElideRight, itemOption.rect.width() - 30 - 10); //当字体超过Item的长度时显示为省略号
|
2021-05-25 19:42:40 +08:00
|
|
|
|
QString htmlString;
|
|
|
|
|
if((indexColString.toUpper()).contains((m_regFindKeyWords.toUpper()))) {
|
|
|
|
|
indexFindLeft = indexColString.toUpper().indexOf(m_regFindKeyWords.toUpper()); //得到查找字体在当前整个Item字体中的位置
|
|
|
|
|
htmlString = escapeHtml(indexColString.left(indexFindLeft)) + "<b>" + escapeHtml(indexColString.mid(indexFindLeft, m_regFindKeyWords.length())) + "</b>" + escapeHtml(indexColString.right(indexColString.length() - indexFindLeft - m_regFindKeyWords.length()));
|
|
|
|
|
} else {
|
|
|
|
|
bool boldOpenned = false;
|
|
|
|
|
for(int i = 0; i < indexColString.length(); i++) {
|
|
|
|
|
if((m_regFindKeyWords.toUpper()).contains(QString(indexColString.at(i)).toUpper())) {
|
|
|
|
|
if(! boldOpenned) {
|
|
|
|
|
boldOpenned = true;
|
|
|
|
|
htmlString.append(QString("<b>"));
|
|
|
|
|
}
|
|
|
|
|
htmlString.append(escapeHtml(QString(indexColString.at(i))));
|
|
|
|
|
} else {
|
|
|
|
|
if(boldOpenned) {
|
|
|
|
|
boldOpenned = false;
|
|
|
|
|
htmlString.append(QString("</b>"));
|
|
|
|
|
}
|
|
|
|
|
htmlString.append(escapeHtml(QString(indexColString.at(i))));
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// qDebug()<<indexColString<<"---->"<<htmlString;
|
2021-12-17 17:39:37 +08:00
|
|
|
|
return "<pre>" + htmlString + "</pre>";
|
2021-05-25 19:42:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString ResultViewDelegate::escapeHtml(const QString &str) const
|
|
|
|
|
{
|
|
|
|
|
QString temp = str;
|
|
|
|
|
temp.replace("<", "<");
|
|
|
|
|
temp.replace(">", ">");
|
|
|
|
|
return temp;
|
|
|
|
|
}
|
2021-12-13 10:53:28 +08:00
|
|
|
|
|
|
|
|
|
ResultItemStyle *ResultItemStyle::getStyle()
|
|
|
|
|
{
|
|
|
|
|
if (!global_instance_of_item_style) {
|
|
|
|
|
global_instance_of_item_style = new ResultItemStyle;
|
|
|
|
|
}
|
|
|
|
|
return global_instance_of_item_style;
|
|
|
|
|
}
|
|
|
|
|
|
2022-05-19 13:47:54 +08:00
|
|
|
|
void ResultItemStyle::drawPrimitive(QStyle::PrimitiveElement element, const QStyleOption *option, QPainter *painter, const QWidget *widget) const
|
|
|
|
|
{
|
|
|
|
|
switch (element) {
|
|
|
|
|
case PE_PanelItemViewItem: {
|
|
|
|
|
|
|
|
|
|
bool isIconView = false;
|
|
|
|
|
auto opt = qstyleoption_cast<const QStyleOptionViewItem *>(option);
|
|
|
|
|
if (!opt)
|
|
|
|
|
return;
|
|
|
|
|
if (opt) {
|
|
|
|
|
isIconView = (opt->decorationPosition & QStyleOptionViewItem::Top);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool isHover = (option->state & State_MouseOver) && (option->state & ~State_Selected);
|
|
|
|
|
bool isSelected = option->state & State_Selected;
|
|
|
|
|
bool enable = option->state & State_Enabled;
|
|
|
|
|
QColor color = option->palette.color(enable? QPalette::Active: QPalette::Disabled, QPalette::Highlight);
|
|
|
|
|
|
|
|
|
|
if (isSelected) {
|
|
|
|
|
color.setAlpha(255);
|
|
|
|
|
} else if (isHover) {
|
|
|
|
|
// color = opt->palette.color(QPalette::Active, QPalette::BrightText);
|
|
|
|
|
// color.setAlpha(0.05);
|
|
|
|
|
color = QColor(241, 241, 241);
|
|
|
|
|
} else {
|
|
|
|
|
color.setAlpha(0);
|
|
|
|
|
}
|
|
|
|
|
QPainterPath path;
|
|
|
|
|
|
|
|
|
|
if(opt->viewItemPosition == QStyleOptionViewItem::OnlyOne) {
|
|
|
|
|
path.addRoundedRect(option->rect, 6, 6);
|
|
|
|
|
} else if(opt->viewItemPosition == QStyleOptionViewItem::Beginning) {
|
|
|
|
|
//一个左侧有两个圆角的矩形
|
|
|
|
|
path.moveTo(option->rect.topLeft() + QPoint(6, 0));
|
|
|
|
|
path.cubicTo(option->rect.topLeft() + QPoint(6, 0),
|
|
|
|
|
option->rect.topLeft(), option->rect.topLeft() +
|
|
|
|
|
QPoint(0, 6));
|
|
|
|
|
|
|
|
|
|
path.lineTo(option->rect.bottomLeft() - QPoint(0, 6));
|
|
|
|
|
path.cubicTo(option->rect.bottomLeft() - QPoint(0, 6),
|
|
|
|
|
option->rect.bottomLeft() + QPoint(0, 1),
|
|
|
|
|
option->rect.bottomLeft() + QPoint(6, 1));
|
|
|
|
|
|
|
|
|
|
path.lineTo(option->rect.bottomRight() + QPoint(1, 1));
|
|
|
|
|
path.lineTo(option->rect.topRight()+ QPoint(1, 0));
|
|
|
|
|
path.lineTo(option->rect.topLeft() + QPoint(6, 0));
|
|
|
|
|
} else if(opt->viewItemPosition == QStyleOptionViewItem::Middle) {
|
|
|
|
|
path.addRect(option->rect.adjusted(-1, 0, 1, 0));
|
|
|
|
|
} else if(opt->viewItemPosition == QStyleOptionViewItem::End) {
|
|
|
|
|
//一个右侧有两个圆角的矩形
|
|
|
|
|
path.moveTo(option->rect.topRight() + QPoint(-6, 0));
|
|
|
|
|
path.cubicTo(option->rect.topRight() + QPoint(-6, 0),
|
|
|
|
|
option->rect.topRight(),
|
|
|
|
|
option->rect.topRight() +
|
|
|
|
|
QPoint(0, 6));
|
|
|
|
|
|
|
|
|
|
path.lineTo(option->rect.bottomRight() - QPoint(0, 6));
|
|
|
|
|
path.cubicTo(option->rect.bottomRight() - QPoint(0, 6),
|
|
|
|
|
option->rect.bottomRight() + QPoint(0, 1),
|
|
|
|
|
option->rect.bottomRight() + QPoint(-6, 1));
|
|
|
|
|
|
|
|
|
|
path.lineTo(option->rect.bottomLeft() + QPoint(0, 1));
|
|
|
|
|
path.lineTo(option->rect.topLeft());
|
|
|
|
|
path.lineTo(option->rect.topRight() + QPoint(-6, 0));
|
|
|
|
|
} else {
|
|
|
|
|
// path.addRoundedRect(option->rect, 8, 8);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
painter->save();
|
|
|
|
|
painter->setRenderHint(QPainter::Antialiasing);
|
|
|
|
|
painter->setPen(Qt::NoPen);
|
|
|
|
|
painter->setBrush(color);
|
|
|
|
|
painter->drawPath(path);
|
|
|
|
|
// painter->fillPath(path, painter->brush());
|
|
|
|
|
|
|
|
|
|
painter->restore();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
return QProxyStyle::drawPrimitive(element, option, painter, widget);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-13 10:53:28 +08:00
|
|
|
|
void ResultItemStyle::drawControl(QStyle::ControlElement element, const QStyleOption *option, QPainter *painter, const QWidget *widget) const
|
|
|
|
|
{
|
|
|
|
|
switch (element) {
|
|
|
|
|
case CE_ItemViewItem: {
|
|
|
|
|
if (const QStyleOptionViewItem *vopt = qstyleoption_cast<const QStyleOptionViewItem *>(option)) {
|
|
|
|
|
painter->save();
|
|
|
|
|
if (painter->clipPath().isEmpty()) {
|
|
|
|
|
painter->setClipRect(option->rect);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QRect checkRect = proxy()->subElementRect(SE_ItemViewItemCheckIndicator, vopt, widget);
|
|
|
|
|
QRect iconRect = proxy()->subElementRect(SE_ItemViewItemDecoration, vopt, widget);
|
|
|
|
|
|
|
|
|
|
// draw the background
|
|
|
|
|
proxy()->drawPrimitive(PE_PanelItemViewItem, option, painter, widget);
|
|
|
|
|
|
|
|
|
|
// draw the check mark
|
|
|
|
|
if (vopt->features & QStyleOptionViewItem::HasCheckIndicator) {
|
|
|
|
|
QStyleOptionViewItem option(*vopt);
|
|
|
|
|
option.rect = checkRect;
|
|
|
|
|
option.state = option.state & ~QStyle::State_HasFocus;
|
|
|
|
|
|
|
|
|
|
switch (vopt->checkState) {
|
|
|
|
|
case Qt::Unchecked:
|
|
|
|
|
option.state |= QStyle::State_Off;
|
|
|
|
|
break;
|
|
|
|
|
case Qt::PartiallyChecked:
|
|
|
|
|
option.state |= QStyle::State_NoChange;
|
|
|
|
|
break;
|
|
|
|
|
case Qt::Checked:
|
|
|
|
|
option.state |= QStyle::State_On;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
proxy()->drawPrimitive(QStyle::PE_IndicatorViewItemCheck, &option, painter, widget);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// draw the icon
|
|
|
|
|
QIcon::Mode mode = QIcon::Normal;
|
|
|
|
|
if (!(vopt->state & QStyle::State_Enabled)) {
|
|
|
|
|
mode = QIcon::Disabled;
|
|
|
|
|
} else if (vopt->state & QStyle::State_Selected) {
|
|
|
|
|
mode = QIcon::Selected;
|
|
|
|
|
}
|
|
|
|
|
QIcon::State state = vopt->state & QStyle::State_Open ? QIcon::On : QIcon::Off;
|
|
|
|
|
auto pixmap = vopt->icon.pixmap(vopt->decorationSize, mode, state);
|
|
|
|
|
|
|
|
|
|
iconRect.moveLeft(8);
|
|
|
|
|
QStyle::drawItemPixmap(painter, iconRect, vopt->decorationAlignment, pixmap);
|
|
|
|
|
painter->restore();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|