QTreeView实现圆角样式

QTreeView实现圆角样式QTreeView实现圆角样式在QTreeView等继承于QAbstractItemView表格中,定制表格样式通常都是通过设置项目代理(ItemDelegate)来实现。在这种实现方法中,每个项目(Item)基本上是孤立的,无法有效判断己身周遭环境。如果以此种方法来实现圆角样式,行首或许还能通过方法intQModelindex::column()来判断是否属于第一列来断定,然而行尾难道还要通过QAbstractItemModelQModelindex::*model()获取模型(Model)之后再

大家好,又见面了,我是你们的朋友全栈君。

QTreeView实现圆角样式

在QTreeView等继承于QAbstractItemView表格中,定制表格样式通常都是通过设置项目代理(ItemDelegate)来实现。在这种实现方法中,每个项目(Item)基本上是孤立的,无法有效判断己身周遭环境。
如果以此种方法来实现圆角样式,行首或许还能通过方法int QModelindex::column()来判断是否属于第一列来断定,然而行尾难道还要通过QAbstractItemModel QModelindex::*model()获取模型(Model)之后再来判断吗?且视图(View)本身可以隐藏、拖拽指定行、列,所以这种方式并不靠谱。
更正:
QStyleOptionViewItem提供了获取可视项目位置的接口,可通过

QStyleOptionViewItem::viewItemPosition

这个枚举值获取。更正此处错误描述并更新

QPainterPath TreeViewStyle::roundedPath(const QStyleOptionViewItem *o, const QWidget *w) const

样式表(StyleSheet)似乎也不支持此种操作。

继承QProxyStyle

为了保持与系统或部件所使用的样式一致,应当选择继承QProxyStyle而非QStyle及其它(QCommonStyle等)。

class TreeViewStyle : public QProxyStyle
{ 
   
	Q_OBJECT
public:
    explicit TreeViewStyle(QStyle *style = nullptr);
}

重载绘制函数

    void drawPrimitive(QStyle::PrimitiveElement pe, const QStyleOption *opt, QPainter *p, const QWidget *w) const override;

绘制选中状态

    void drawPanelItemViewRow(const QStyleOption *opt, QPainter *p, const QWidget *w) const;

绘制行背景

行背景即列表背景,通常情况下没有绘制。一般仅在设置了行颜色交替的情况下绘制。

    void drawPanelItemViewItem(const QStyleOption *opt, QPainter *p, const QWidget *w) const;

绘制路径计算函数

计算绘制路径即选中状态、行背景所需绘制区域。

    QPainterPath roundedPath(const QStyleOptionViewItem *opt, const QWidget *w) const;

用法

QTreeView view = new QTreeView(this);
view->setStyle(new TreeViewStyle(view->style()));

效果

圆角样式的QTreeView,选中行的色块与颜色交替行的色块四角都是圆角

相关代码

TreeViewStyle.h

#ifndef TREEVIEWSTYLE_H
#define TREEVIEWSTYLE_H

#include <QProxyStyle>

class QStyleOptionViewItem;
class TreeViewStyle : public QProxyStyle
{ 
   
    Q_OBJECT
public:
    explicit TreeViewStyle(QStyle *style = nullptr);

    void drawPrimitive(QStyle::PrimitiveElement pe, const QStyleOption *opt, QPainter *p, const QWidget *w) const override;
    QPainterPath roundedPath(const QStyleOptionViewItem *opt, const QWidget *w) const;
    void drawPanelItemViewRow(const QStyleOption *opt, QPainter *p, const QWidget *w) const;
    void drawPanelItemViewItem(const QStyleOption *opt, QPainter *p, const QWidget *w) const;
};

#endif // TREEVIEWSTYLE_H

TreeViewStyle.cpp

#include "TreeViewStyle.h"

#include <QStyleOption>
#include <QPainter>
#include <DStyle> //仅用于获取部件设置的圆角大小,非Dtk用户请删除此行

TreeViewStyle::TreeViewStyle(QStyle *style) : QProxyStyle(style)
{ 
   

}

void TreeViewStyle::drawPrimitive(QStyle::PrimitiveElement pe, const QStyleOption *opt, QPainter *p, const QWidget *w) const
{ 
   
    switch (pe) { 
   
    case PE_PanelItemViewRow:
        drawPanelItemViewRow(opt, p, w);
        break;
    case PE_PanelItemViewItem:
        drawPanelItemViewItem(opt, p, w);
        break;
    default:
        QProxyStyle::drawPrimitive(pe, opt, p, w);
        break;
    }
}

QPainterPath TreeViewStyle::roundedPath(const QStyleOptionViewItem *o, const QWidget *w) const
{ 
   
    int radius = Dtk::Widget::DStyle::pixelMetric(this, Dtk::Widget::DStyle::PM_FrameRadius, o, w); 
    //用于获取部件设置的圆角大小,非Dtk用户请直接设置radius的值,Dtk中默认为8
    QPainterPath path;
    path.setFillRule(Qt::WindingFill);
    QRect corner(0, 0, radius, o->rect.height());
    switch (o->viewItemPosition) { 
   
    case QStyleOptionViewItem::Beginning:
        path.addRoundedRect(o->rect, radius, radius);
        corner.moveTopRight(o->rect.topRight());
        path.addRect(corner);
        break;
    case QStyleOptionViewItem::End:
        path.addRoundedRect(o->rect, radius, radius);
        corner.moveTopLeft(o->rect.topLeft());
        path.addRect(corner);
        break;
    case QStyleOptionViewItem::OnlyOne:
        path.addRoundedRect(o->rect, radius, radius);
        break;
    case QStyleOptionViewItem::Middle:
        path.addRect(o->rect);
        break;
    case QStyleOptionViewItem::Invalid:
        break;
    }
// path.addRoundedRect(o->rect, radius, radius);
// QRect corner;
// corner.setSize({radius, radius});
// if (o->rect.right() != w->rect().right()) { 
   
// corner.moveTopRight(o->rect.topRight());
// path.addRect(corner);
// corner.moveBottomRight(o->rect.bottomRight());
// path.addRect(corner);
// }
// if (o->rect.left() != w->rect().left()) { 
   
// corner.moveTopLeft(o->rect.topLeft());
// path.addRect(corner);
// corner.moveBottomLeft(o->rect.bottomLeft());
// path.addRect(corner);
// }
    return path;
}

void TreeViewStyle::drawPanelItemViewRow(const QStyleOption *opt, QPainter *p, const QWidget *w) const
{ 
   
    p->save();
    if (const QStyleOptionViewItem *vopt = qstyleoption_cast<const QStyleOptionViewItem *>(opt)) { 
   
        p->setRenderHint(QPainter::Antialiasing);
        QPainterPath path = roundedPath(vopt, w);
        QPalette::ColorGroup cg = (w ? w->isEnabled() : (vopt->state & QStyle::State_Enabled))
                                  ? QPalette::Normal : QPalette::Disabled;
        if (cg == QPalette::Normal && !(vopt->state & QStyle::State_Active))
            cg = QPalette::Inactive;

        if ((vopt->state & QStyle::State_Selected) &&  proxy()->styleHint(QStyle::SH_ItemView_ShowDecorationSelected, opt, w))
            p->fillRect(vopt->rect, vopt->palette.brush(cg, QPalette::Highlight));
        else if (vopt->features & QStyleOptionViewItem::Alternate)
            p->fillPath(path, vopt->palette.brush(cg, QPalette::AlternateBase));
    }
    p->restore();
}

void TreeViewStyle::drawPanelItemViewItem(const QStyleOption *opt, QPainter *p, const QWidget *w) const
{ 
   
    p->save();
    if (const QStyleOptionViewItem *vopt = qstyleoption_cast<const QStyleOptionViewItem *>(opt)) { 
   
        p->setRenderHint(QPainter::Antialiasing);
        QPainterPath path = roundedPath(vopt, w);
        QPalette::ColorGroup cg = (w ? w->isEnabled() : (vopt->state & QStyle::State_Enabled))
                                  ? QPalette::Normal : QPalette::Disabled;
        if (cg == QPalette::Normal && !(vopt->state & QStyle::State_Active))
            cg = QPalette::Inactive;
        if (vopt->showDecorationSelected && (vopt->state & QStyle::State_Selected)) { 
   
            p->fillPath(path, vopt->palette.brush(cg, QPalette::Highlight));
        } else { 
   
            if (vopt->backgroundBrush.style() != Qt::NoBrush) { 
   
                QPointF oldBO = p->brushOrigin();
                p->setBrushOrigin(vopt->rect.topLeft());
                p->fillPath(path, vopt->backgroundBrush);
                p->setBrushOrigin(oldBO);
            }

            if (vopt->state & QStyle::State_Selected) { 
   
                QRect textRect = subElementRect(QStyle::SE_ItemViewItemText,  opt, w);
                p->fillRect(textRect, vopt->palette.brush(cg, QPalette::Highlight));
            }
        }
    }
    p->restore();
}

联系方式

QQ群:236510798

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/134345.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • JS和H5做一个音乐播放器,附带源码

    JS和H5做一个音乐播放器,附带源码

    2021年10月10日
    115
  • Python科学计算之Pandas[通俗易懂]

    Python科学计算之Pandas[通俗易懂]Reference: http://mp.weixin.qq.com/s?src=3&timestamp=1474979163&ver=1&signature=wnZ

    2022年7月5日
    22
  • android bindservice方法,android BindService[通俗易懂]

    android bindservice方法,android BindService[通俗易懂]1、绑定服务1.1绑定方式通过服务链接(ServiceConnection)或直接获取Service中状态和数据信息服务链接能够获取Service的对象,因此绑定Service的组件可以调用Service中的实现的函数使用Service的组件通过Context.bindService()建立服务链接,通过Context.unbindService()停止服务链接如果在绑定过程中Service没有启…

    2022年5月1日
    190
  • jxl.jar下载

    jxl.jar下载jxl.jar给java提供了简单操作Excel的方法:链接:https://pan.baidu.com/s/17HXj_w8E2nM8iIssf2Bmhg提取码:l6t5

    2022年7月26日
    4
  • 记忆化搜索的研究

    记忆化搜索的研究记忆化搜索:算法上依然是搜索的流程,但是搜索到的一些解用动态规划的那种思想和模式作一些保存。一般说来,动态规划总要遍历所有的状态,而搜索可以排除一些无效状态。更重要的是搜索还可以剪枝,可能剪去大量不必要的状态,因此在空间开销上往往比动态规划要低很多。记忆化算法在求解的时候还是按着自顶向下的顺序,但是每求解一个状态,就将它的解保存下来,以后再次遇到这个状态的时候,就不必重新求解了。这种方法综合了搜索和动态规划两方面的优点,因而还是很有实用价值的。上传/更换附件动态规划的另一种实现形式——记忆化搜索的应用

    2022年7月26日
    11
  • networkmanager详解_networkmanager是什么服务

    networkmanager详解_networkmanager是什么服务NetworkManager服务是管理和监控网络设置的守护进程,CentOS7更加注重使用NetworkManager服务来实现网络的配置和管理,CentOS7以前是通过network服务管理网络,以后的版本所有网络管理和设置统一由NetworkManager服务来维护。它是一个动态的,事件驱动的网络管理服务。下面看一下CentOS7中network和NetworkManger的信息吧。注意:…

    2022年10月4日
    0

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号