QRowTable表格控件重写表头排序支持第三次单击恢复默认排序
一、原生表格
开发客户端程序的方式越来越多了,现在很流行的libcef、electron等等都可以作为快速开发客户端软件的方案,但是如果需要一个好的用户体验,还是离不开原生化的开发,虽然慢,但是性能好啊。
说到原生化开发,那对应的UI库相对较多,流行的就有Qt、soui、duilib、还有老掉牙的MFC和其他一些第三方公司开源维护的directUI库等等。网上找到一篇整理的文章,有兴趣的同学可以参考C++界面库。
目前C++客户端使用最广泛的就是UI框架就是Qt,它不仅包含了GUI控件,更多的其实是一种解决方案,使用过Qt的同学都比较清楚,Qt的安装动态库有几十个之多,可是如果你只想使用Qt的GUi模块的话,就只需要包含3个动态库即可,他们分别是核心模块QtCore、Gui模块QtGui和QtWidget。
当我们将Qt作为我们的开发SDK时,大多数时候原生的控件+qss美化就可以完成我们的需求,然而总有一些特殊情况,比如产品经理脑残的时候、或者说是业务场景傻逼了,总是需要完成一些奇奇怪怪交互,那么我们就需要重写原生的控件实现方式。
今天就来说一个案例--表格控件列排序方式。
对于某一些特殊的场景下,我们的表格展示的数据可能需要排序,这样的表格控件Qt已经给我们提供好了,只需要我们从写一些类和接口即可。前边已经写了4篇关于表格控件的功能,分别是QRowTable表格控件-支持hover整行、checked整行、指定列排序等、QRowTable表格控件(二)-红涨绿跌、QRowTable表格控件(三)-效率优化之-合理使用QStandardItem和QRowTable表格控件(四)-效率优化之-优化数据源,感兴趣的同学可以去看看。
不幸的是Qt自带的表格排序功能,即是我们冲写完所有接口依然不能满足负责的业务需求--表格表头连续3点三次是一个循环,什么意思呢?
传统表格排序
Qt自带的表格排序行为是这样的,默认情况是不排序的,我们可以通过接口开启排序,或者通过接口设置不排序,当我们启用排序规则后,假设说我们的表格点击点击一下是降序,点击第二下就是升序,再次点击是又会恢复到降序规则,这样是不是还挺完美呢!
这个时候产品有话说了,点击第三次时,需要设置程序为不排序。
程序员:卧槽…你说什么…听不到…
【领QT开发教程 学习资料,点击下方链接免费领取↓↓ ,先码住不迷路~】「链接」
新的排序规则
在传统表格排序的基础上做一下调整 可以支持某些列不允许排序 3次连续单击为一个循环,也就是降序-升序-无序这样3个状态循环
不得不说产品的脑洞还是很大的。既然产品都说想要这个功能了,那么有时间还是得考虑下。跟产品沟通良久后,有了如下安排,虽然这个功能对用户来说不是一个特别需要的功能,但是当我们的软件功能稳定后,迭代没有那么着急的时候,是不是可以考虑研究下这个而功能呢。
然后也就有了本篇文章
虽然重写了Qt本身的逻辑,没想到还是可以实现的!!!二、效果展示
按照惯例先上图,看看是不是同学们想要的功能。
三、实现方式
新的表格排序有2个点需要我们去思考,分别是某些列排序、排序交互修改,下面我们分开来去实现 1、排序列定制
Qt默认提供了可排序接口,但是开启后我们所有的列都支持排序了,这个时候我们就需要研究Qt的源码,看看Qt的排序是怎么触发的,然后在合理的时机去加上我们不支持排序的逻辑
博主这里找到的做法是重写QHaderView这个类,并重写实现了鼠标按下并抬起的接口,在这个接口中判断我们业务层是不是允许排序。
重写后的逻辑是这样的 如果不允许排序,我们先调用Qt原有的接口禁用所有列排序功能 调用父类的鼠标抬起函数 如果不允许排序,需要调用Qt原有的接口启用所有列排序功能
以上逻辑的步骤2是原有的逻辑,步骤1和步骤3分别是在进行业务逻辑判断后进行的逻辑调整,达到我们禁用某些列排序的功能 void QRowHeader::mouseReleaseEvent(QMouseEvent * event) { int column = logicalIndexAt(event->pos().x()); if (m_Indicator.contains(column) && m_Indicator[column] == false && qAbs(event->x() - m_pPressPos.x()) < QApplication::startDragDistance()/*非拖拽*/) { setSectionsClickable(false); QHeaderView::mouseReleaseEvent(event); setSectionsClickable(true); } QHeaderView::mouseReleaseEvent(event); ...
如上代码,m_Indicator结构中存储了我们想要定制的是否允许排序的列,只需要对外暴露一个修改接口即可,代码是不是还很简单呢。 void QRowHeader::SetSortEnable(int logicalIndex, bool enable) { m_Indicator[logicalIndex] = enable; }
这样禁用某些列排序的功能就是实现了。
禁用某些列排序这个功能相对来说好实现一些,毕竟重写的逻辑不是特别复杂,下面我们来看下怎么修改排序交互逻辑。 2、排序交互修改
有了前面的问题处理思路,这个功能依然如此,我们先跟Qt源码,看看已有的排序交互逻辑实现怎么实现的,然后在合理的实际去重置某些变量的值,达到重写交互逻辑。
博主这里跟踪完Qt默认的排序实现方式后,发现重写这个功能还是有一定难度的,首先数据排序是一块,另一块是一旦启用排序后,排序箭头的绘制这块也需要去重写。
什么意思呢?
Qt的排序规则一旦启用,排序的箭头就会被绘制,并且绘制箭头的逻辑还简单粗暴,不是升序就是降序。
【领QT开发教程 学习资料,点击下方链接免费领取↓↓ ,先码住不迷路~】「链接」
且看如下Qt表头绘制源码
如果显示排序箭头并且是排序列,则需要绘制排序箭头,不是降序就是升序
我:我槽…什么鬼,就不能提供一个空的枚举吗? void QHeaderView::paintSection(QPainter *painter, const QRect &rect, int logicalIndex) const { ... if (isSortIndicatorShown() && sortIndicatorSection() == logicalIndex) opt.sortIndicator = (sortIndicatorOrder() == Qt::AscendingOrder) ? QStyleOptionHeader::SortDown : QStyleOptionHeader::SortUp; ... }
既然这条路子是走不通了,那么我们只能取巧,换其他方式。
终于、皇天不负有心人被博主想到一个好办法,代码量依然还是很少。
回到我们最开始的需求,我们其实就是想让第三次点击时,只是一个特殊的操作,然后把第四点击当做第一次单击即可。
有了这个想法后,那么就来干吧,既然还是重写鼠标抬起函数,想尽一切办法监控连续的第三次单击,把他处理成非排序状态,然后在下一次单击时,走正常的排序逻辑。 void QRowHeader::mouseReleaseEvent(QMouseEvent * event) { ... QHeaderView::mouseReleaseEvent(event); column = logicalIndexAt(event->pos().x()); static bool nextNoSort = false; static int prevColumn = -1; if (nextNoSort && prevColumn == column && qAbs(event->x() - m_pPressPos.x()) < QApplication::startDragDistance()) { emit RestoreSort();//恢复默认排序 } if (nextNoSort != true && prevColumn == column && sectionsClickable() && sortIndicatorOrder() == Qt::DescendingOrder) { if (qAbs(event->x() - m_pPressPos.x()) < QApplication::startDragDistance()) { nextNoSort = true; prevColumn = column; } else { nextNoSort = false; } } else { if (nextNoSort && qAbs(event->x() - m_pPressPos.x()) >= QApplication::startDragDistance()) { nextNoSort = true; } else { nextNoSort = false; prevColumn = column; } } ... }
好了,代码以上。不过这里重点还是要说下为什么这么干!
首先需要调用父类的mouseReleaseEvent函数,否则拖拽会有问题,而且必须调用这个函数才可以让内存数据正常。
然后我们记录了一系列内存状态,判断是不是需要恢复排序状态,当条件满足时发出RestoreSort信号,外部程序只需要接收这个信号,然后恢复默认排序即可。
恢复默认排序 void QRowTable::RestoreSort() { m_pFilter->SetCompareType(QFilterModel::CT_NULL); m_pFilter->invalidate(); m_pHHeader->setSortIndicator(-1, Qt::DescendingOrder); }
购买茅台1935,贪图便宜吃大亏人红是非多,酒红假酒多。2023年3月15日,茅台酱香酒打假知识产权保护部在一个内部会议上透露,近期他们分别在拼XXX宝X东等网络电商平台上以不高于1100元瓶的价格,累计购买了1
育儿假,军人家庭真的很需要!剑客北海三剑客三剑客题图大唐文图来源于网络剑客君后台经常收到各类的问题和求助,关于军人家庭方面,育儿假真的被问的很多。还记得曾有人跟剑客君说如果军人哺乳期育儿假不能出台好消息,实在
央企估值修复新逻辑ESG推动财务绩效稳健提升国企特别是央企是社会发展的中坚力量,ESG作为可持续发展理念,近年愈加受到国内机构投资者重视。当央企进行ESG实践走向深入,会带来那些积极影响。周二,华宝证券在报告中分析了ESG实
完美日记母公司上市以来首次单季盈利,护肤品总营收逼近彩妆近日,逸仙电商发布2022年第四季度及全年财务业绩报告,公告显示,公司第四季度实现营收10。1亿元,同比下降34。2净亏损为5500万元,与去年同期相比收窄了88。4。NonGAA
GPT4全网首个测评来了!逻辑性创造性,你想知道的我们都测了3月15日凌晨1点,OpenAI宣布推出GPT4,在功能和一致性方面取得了前所未有的成果。(官网链接httpsopenai。comproductgpt4)公众号后台回复GPT4可获
逼孩子学习有用吗?先弄清学习的底层逻辑!这是一本适合家长和教师看的书,能让人们了解孩子喜欢与不喜欢上学的真实原因和底层逻辑。书中每一章都提出一个概念,分析对学习的影响,解释相关原因,并总结对课堂的启示,是一本对教室上课和
17岁黄花闺女嫁入54岁空军上将,为他先后怀孕16次,后来怎样?不论什么年代,17岁,是一位女子最美好的年龄,也是每个少女都会对爱情充满向往,在她们的想象中,自己的意中人一定是个顶天立地的男子汉。而且有时候爱情之间的年龄差真的很重要吗?在民国时
刚刚!两会后第二只部级大老虎接受审查,曾担任首都组织部部长大快人心,一大早就听到反腐的好消息,今天的主角可大有来头,如假包换的大老虎。3月17日,据权威媒体报道,宁夏回族自治区党委原副书记,银川市委原书记姜志刚接受调查。看照片,长得文质彬
神兵奇迹手游攻略神兵奇迹手游弓箭手怎么加点弓箭手到底该如何加点?弓箭手到底如何去玩,如何去加点?本次就由小编来给大家仔细的分析一波,对于新手玩家来说一篇好的攻略,还是能起到不错的效果的,下面就一起来看看怎么玩吧!还有就是最
因地制宜建设富美乡村虽是工作日,广东省东莞市茶山镇南社明清古村落一天接待了五六个旅游团。今年春节,村里的醒狮游园非遗美食节游船唱粤剧等传统文化活动每天都吸引上万名游客接下来的清明踏青游园活动,预计又将
大西北环线之仁康古屋2017年10月24日下午到达四川省甘孜州的世界高城理塘,这里有七世达赖格桑嘉措的旧居仁康古屋。说藏族中流传着这么一种说法,如果经过理塘去西藏朝圣,没有到仁康古屋朝拜的,就等于没有