一、前言

上次在写大屏数据可视化电子看板系统时候,提到过改造QCustomPlot来实现柱状分组图、横向柱状图、横向分组图、鼠标悬停提示等。这次单独列出来描述,有很多人疑问为啥不用QChart,或者echart等形式,其实这两种方式我都尝试过,比如Qt5.7以后新增的QChart模块,曲线这块,支持数据量很小,而且用法极其不适应,非常别扭,尤其是10W以上数据量的支持,简直是渣渣,优点也是有很多的,比如动画效果,我看过他的完整源码,动画这块处理的非常好,连坐标轴都可以有动画效果,而且支持很多种效果,而且内置了很多套theme皮肤,省去了很多渣渣审美的程序员自己来配色,这个倒是挺方便的。而对于echart,必须依赖浏览器控件,资源占用比较高,后面决定采用改造QCustomPlot来实现用户需要的各种图表效果。

在整个改造的过程中,全部封装成易用的函数,传入参数即可,同时还支持全局样式更改,支持样式表控制整体颜色更改,考虑了很多细节,比如弹出悬停信息的位置等,都自动计算显示在最佳最合理位置。考虑到很多人用的QCustomPlot1.0,特意还做了QCustomPlot1.0和2.0的完全兼容。

二、实现的功能

  • 1:可设置X轴Y轴范围值
  • 2:可设置背景颜色+文本颜色+网格颜色
  • 3:可设置三条曲线颜色+颜色集合
  • 4:可设置是否显示定位十字线,可分别设置横向和纵向
  • 5:可设置十字线的宽度和颜色
  • 6:可设置是否显示数据点以及数据点的大小
  • 7:可设置是否填充背景形成面积图
  • 8:可设置模式-拖动+缩放等
  • 9:可设置坐标轴间距+第二坐标系可见
  • 10:提供接口setDataLine直接设置曲线,支持多条
  • 11:提供接口setDataBar直接设置柱状图,支持多条形成堆积图
  • 12:提供接口setLabs设置文本标签替代key
  • 13:提供清空+重绘接口+外部获取QCustomPlot对象
  • 14:提供函数start+stop来模拟正弦曲线
  • 15:可设置柱状图的值的位置+精确度+颜色
  • 16:支持鼠标移动到数据点高亮显示数据点以及显示数据提示信息
  • 17:可设置提示信息位置 自动处理+顶部+右上角+右侧+右下角+底部+左下角+左侧+左上角
  • 18:可设置是否校验数据产生不同的背景颜色,比如柱状图的每根柱子都可以根据数据生成不同背景颜色
  • 19:可设置是否显示图例+图例位置+图例行数
  • 20:支持多条曲线+柱状图+柱状分组图+横向柱状图+横向柱状分组图+柱状堆积图
  • 21:内置15套精美颜色,自动取颜色集合的颜色,省去配色的烦恼
  • 22:同时支持 QCustomPlot 1.0 和 QCustomPlot 2.0

三、效果图

四、核心代码

  1. void CustomPlot::setDataLine(int index, const QString &name, const QVector<double> &key, const QVector<double> &value)
  2. {
  3. if (customPlot->graphCount() > index) {
  4. customPlot->graph(index)->setName(name);
  5. customPlot->graph(index)->setData(key, value);
  6. customPlot->xAxis->setRange(-offsetX, key.count() + offsetX, Qt::AlignLeft);
  7. //超过3条线条颜色设置颜色集合的颜色
  8. if (index >= 3) {
  9. setColor(index, colors.at(index));
  10. } else {
  11. setColor(0, colors.at(0));
  12. setColor(1, colors.at(1));
  13. setColor(2, colors.at(2));
  14. }
  15. }
  16. }
  17. void CustomPlot::setDataBarv(const QStringList &rowNames,
  18. const QStringList &columnNames,
  19. const QList<QVector<double> > &values,
  20. const QColor &borderColor,
  21. int valuePosition,
  22. int valuePrecision,
  23. const QColor &valueColor,
  24. bool checkData)
  25. {
  26. //只有1列的才能设置
  27. if (columnNames.count() != 1) {
  28. return;
  29. }
  30. //可以直接用堆积图,因为只有一列的柱状图不会形成堆积
  31. setDataBars(rowNames, columnNames, values, borderColor, valuePosition, valuePrecision, valueColor, checkData);
  32. }
  33. void CustomPlot::setDataBarvs(const QStringList &rowNames,
  34. const QStringList &columnNames,
  35. const QList<QVector<double> > &values,
  36. const QColor &borderColor,
  37. int valuePosition,
  38. int valuePrecision,
  39. const QColor &valueColor,
  40. bool checkData)
  41. {
  42. //过滤个数不一致数据,防止索引越界
  43. int rowCount = rowNames.count();
  44. int columnCount = columnNames.count();
  45. int valueCount = values.count();
  46. if (columnCount == 0 || valueCount == 0 || columnCount != valueCount) {
  47. return;
  48. }
  49. //设置网格线不显示,会更好看
  50. customPlot->xAxis->grid()->setVisible(false);
  51. //customPlot->yAxis->grid()->setVisible(false);
  52. //设置横坐标文字描述
  53. QVector<double> ticks;
  54. QVector<QString> labels;
  55. int count = rowCount * columnCount;
  56. for (int i = 0; i < rowCount; i++) {
  57. ticks << 1.5 + (i * columnCount);
  58. labels << rowNames.at(i);
  59. }
  60. setLabX(ticks, labels);
  61. customPlot->xAxis->setRange(0, count + 1);
  62. for (int i = 0; i < columnCount; i++) {
  63. //同样也要先过滤个数是否符合要求
  64. QVector<double> value = values.at(i);
  65. if (rowCount != value.count()) {
  66. continue;
  67. }
  68. //创建柱状图
  69. CustomBarv *bar = new CustomBarv(customPlot->xAxis, customPlot->yAxis);
  70. bar->setCheckData(checkData);
  71. //设置宽度比例
  72. bar->setWidth(0.9);
  73. //设置显示值的位置 0-不绘制 1-顶部上面 2-顶部居中 3-中间居中 4-底部居中
  74. bar->setValuePostion(valuePosition);
  75. bar->setValuePrecision(valuePrecision);
  76. bar->setValueColor(valueColor);
  77. //设置名称
  78. bar->setName(columnNames.at(i));
  79. //设置颜色,取颜色集合
  80. QColor color = QColor(51, 204, 255);
  81. if (i < colors.count()) {
  82. color = colors.at(i);
  83. }
  84. //边缘高亮,如果传入了边框颜色则取边框颜色
  85. bar->setPen(QPen(borderColor == Qt::transparent ? color.light(150) : borderColor));
  86. bar->setBrush(color);
  87. //这个算法很巧妙,想了很久
  88. QVector<double> ticks;
  89. double offset = i * 0.9;
  90. for (int j = 0; j < rowCount; j++) {
  91. ticks << 1.0 + (j * columnCount) + offset;
  92. }
  93. //设置数据
  94. bar->setData(ticks, value);
  95. }
  96. }
  97. void CustomPlot::setDataBarh(const QStringList &rowNames,
  98. const QStringList &columnNames,
  99. const QList<QVector<double> > &values,
  100. const QColor &borderColor,
  101. int valuePosition,
  102. int valuePrecision,
  103. const QColor &valueColor,
  104. bool checkData)
  105. {
  106. //只有1列的才能设置
  107. if (columnNames.count() != 1) {
  108. return;
  109. }
  110. //过滤个数不一致数据,防止索引越界
  111. int rowCount = rowNames.count();
  112. int columnCount = columnNames.count();
  113. int valueCount = values.count();
  114. if (columnCount == 0 || valueCount == 0 || columnCount != valueCount) {
  115. return;
  116. }
  117. //设置网格线不显示,会更好看
  118. customPlot->xAxis->grid()->setVisible(false);
  119. customPlot->yAxis->grid()->setVisible(false);
  120. customPlot->yAxis->setTickLength(0, 0);
  121. //设置横坐标文字描述
  122. QVector<double> ticks;
  123. QVector<QString> labels;
  124. int count = rowCount * columnCount;
  125. double padding = 1;
  126. for (int i = 0; i < rowCount; i++) {
  127. ticks << padding + (i * columnCount);
  128. labels << rowNames.at(i);
  129. }
  130. setLabY(ticks, labels);
  131. customPlot->yAxis->setRange(0, count + 1);
  132. //先计算出每个柱子占用的高度
  133. double barHeight = 0.7;
  134. for (int i = 0; i < columnCount; i++) {
  135. //同样也要先过滤个数是否符合要求
  136. QVector<double> value = values.at(i);
  137. if (rowCount != value.count()) {
  138. continue;
  139. }
  140. //先绘制系列1的数据,再绘制系列2,依次类推
  141. for (int j = 0; j < rowCount; j++) {
  142. //创建横向柱状图
  143. double y = (0.67 + (j * columnCount));
  144. CustomBarh *bar = new CustomBarh(customPlot);
  145. bar->setCheckData(checkData);
  146. bar->setRect(QPointF(0, y), QPointF(value.at(j), y + barHeight));
  147. bar->setValue(value.at(j));
  148. //设置显示值的位置 0-不绘制 1-顶部上面 2-顶部居中 3-中间居中 4-底部居中
  149. bar->setValuePostion(valuePosition);
  150. bar->setValuePrecision(valuePrecision);
  151. bar->setValueColor(valueColor);
  152. //设置颜色,取颜色集合
  153. QColor color = QColor(51, 204, 255);
  154. if (i < colors.count()) {
  155. color = colors.at(i);
  156. }
  157. //边缘高亮,如果传入了边框颜色则取边框颜色
  158. bar->setPen(QPen(borderColor == Qt::transparent ? color.light(150) : borderColor));
  159. bar->setBrush(color);
  160. }
  161. }
  162. }
  163. void CustomPlot::setDataBarhs(const QStringList &rowNames,
  164. const QStringList &columnNames,
  165. const QList<QVector<double> > &values,
  166. const QColor &borderColor,
  167. int valuePosition,
  168. int valuePrecision,
  169. const QColor &valueColor,
  170. bool checkData)
  171. {
  172. //过滤个数不一致数据,防止索引越界
  173. int rowCount = rowNames.count();
  174. int columnCount = columnNames.count();
  175. int valueCount = values.count();
  176. if (columnCount == 0 || valueCount == 0 || columnCount != valueCount) {
  177. return;
  178. }
  179. //设置网格线不显示,会更好看
  180. customPlot->xAxis->grid()->setVisible(false);
  181. customPlot->yAxis->grid()->setVisible(false);
  182. customPlot->yAxis->setTickLength(0, 0);
  183. customPlot->xAxis->setVisible(false);
  184. //设置横坐标文字描述
  185. QVector<double> ticks;
  186. QVector<QString> labels;
  187. int count = rowCount * columnCount;
  188. //这个算法想了很久,很牛逼
  189. double padding = 1.5 + (columnCount - 2) * 0.4;
  190. for (int i = 0; i < rowCount; i++) {
  191. ticks << padding + (i * columnCount);
  192. labels << rowNames.at(i);
  193. }
  194. setLabY(ticks, labels);
  195. customPlot->yAxis->setRange(0, count + 1);
  196. //先计算出每个柱子占用的高度
  197. double barHeight = 0.8;
  198. for (int i = 0; i < columnCount; i++) {
  199. //同样也要先过滤个数是否符合要求
  200. QVector<double> value = values.at(i);
  201. if (rowCount != value.count()) {
  202. continue;
  203. }
  204. //先绘制系列1的数据,再绘制系列2,依次类推
  205. for (int j = 0; j < rowCount; j++) {
  206. //创建横向柱状图
  207. double y = (0.7 + i * barHeight + (j * columnCount));
  208. CustomBarh *bar = new CustomBarh(customPlot);
  209. bar->setCheckData(checkData);
  210. bar->setRect(QPointF(0, y), QPointF(value.at(j), y + barHeight));
  211. bar->setValue(value.at(j));
  212. //设置显示值的位置 0-不绘制 1-顶部上面 2-顶部居中 3-中间居中 4-底部居中
  213. bar->setValuePostion(valuePosition);
  214. bar->setValuePrecision(valuePrecision);
  215. bar->setValueColor(valueColor);
  216. //设置颜色,取颜色集合
  217. QColor color = QColor(51, 204, 255);
  218. if (j < colors.count()) {
  219. color = colors.at(j);
  220. }
  221. //边缘高亮,如果传入了边框颜色则取边框颜色
  222. bar->setPen(QPen(borderColor == Qt::transparent ? color.light(150) : borderColor));
  223. bar->setBrush(color);
  224. }
  225. }
  226. }
  227. void CustomPlot::setDataBars(const QStringList &rowNames,
  228. const QStringList &columnNames,
  229. const QList<QVector<double> > &values,
  230. const QColor &borderColor,
  231. int valuePosition,
  232. int valuePrecision,
  233. const QColor &valueColor,
  234. bool checkData)
  235. {
  236. //过滤个数不一致数据,防止索引越界
  237. int rowCount = rowNames.count();
  238. int columnCount = columnNames.count();
  239. int valueCount = values.count();
  240. if (columnCount == 0 || valueCount == 0 || columnCount != valueCount) {
  241. return;
  242. }
  243. //设置网格线不显示,会更好看
  244. customPlot->xAxis->grid()->setVisible(false);
  245. //customPlot->yAxis->grid()->setVisible(false);
  246. //先清空原有柱状图
  247. bars.clear();
  248. //设置横坐标文字描述
  249. QVector<double> ticks;
  250. QVector<QString> labels;
  251. for (int i = 0; i < rowCount; i++) {
  252. ticks << i + 1;
  253. labels << rowNames.at(i);
  254. }
  255. setLabX(ticks, labels);
  256. customPlot->xAxis->setRange(0, rowCount + 1);
  257. for (int i = 0; i < columnCount; i++) {
  258. //同样也要先过滤个数是否符合要求
  259. QVector<double> value = values.at(i);
  260. if (rowCount != value.count()) {
  261. continue;
  262. }
  263. //创建柱状堆积图
  264. CustomBarv *bar = new CustomBarv(customPlot->xAxis, customPlot->yAxis);
  265. bar->setCheckData(checkData);
  266. //设置宽度比例
  267. bar->setWidth(0.6);
  268. //设置显示值的位置 0-不绘制 1-顶部上面 2-顶部居中 3-中间居中 4-底部居中
  269. bar->setValuePostion(valuePosition);
  270. bar->setValuePrecision(valuePrecision);
  271. bar->setValueColor(valueColor);
  272. #ifndef old
  273. //设置堆积间隙
  274. if (borderColor != Qt::transparent) {
  275. bar->setStackingGap(1);
  276. }
  277. #endif
  278. //设置名称
  279. bar->setName(columnNames.at(i));
  280. //设置颜色,取颜色集合
  281. QColor color = QColor(51, 204, 255);
  282. if (i < colors.count()) {
  283. color = colors.at(i);
  284. }
  285. //边缘高亮,如果传入了边框颜色则取边框颜色
  286. if (columnCount > 1 && borderColor == Qt::transparent) {
  287. bar->setPen(Qt::NoPen);
  288. } else {
  289. bar->setPen(QPen(borderColor == Qt::transparent ? color.light(150) : borderColor));
  290. }
  291. bar->setBrush(color);
  292. //设置堆积层叠顺序,后面那个移到前一个上面
  293. bars << bar;
  294. if (i > 0) {
  295. bar->moveAbove(bars.at(i - 1));
  296. }
  297. //设置数据
  298. bar->setData(ticks, value);
  299. }
  300. }

五、控件介绍

  1. 超过146个精美控件,涵盖了各种仪表盘、进度条、进度球、指南针、曲线图、标尺、温度计、导航条、导航栏,flatui、高亮按钮、滑动选择器、农历等。远超qwt集成的控件数量。
  2. 每个类都可以独立成一个单独的控件,零耦合,每个控件一个头文件和一个实现文件,不依赖其他文件,方便单个控件以源码形式集成到项目中,较少代码量。qwt的控件类环环相扣,高度耦合,想要使用其中一个控件,必须包含所有的代码。
  3. 全部纯Qt编写,QWidget+QPainter绘制,支持Qt4.6到Qt5.12的任何Qt版本,支持mingw、msvc、gcc等编译器,支持任意操作系统比如windows+linux+mac+嵌入式linux等,不乱码,可直接集成到Qt Creator中,和自带的控件一样使用,大部分效果只要设置几个属性即可,极为方便。
  4. 每个控件都有一个对应的单独的包含该控件源码的DEMO,方便参考使用。同时还提供一个所有控件使用的集成的DEMO。
  5. 每个控件的源代码都有详细中文注释,都按照统一设计规范编写,方便学习自定义控件的编写。
  6. 每个控件默认配色和demo对应的配色都非常精美。
  7. 超过130个可见控件,6个不可见控件。
  8. 部分控件提供多种样式风格选择,多种指示器样式选择。
  9. 所有控件自适应窗体拉伸变化。
  10. 集成自定义控件属性设计器,支持拖曳设计,所见即所得,支持导入导出xml格式。
  11. 自带activex控件demo,所有控件可以直接运行在ie浏览器中。
  12. 集成fontawesome图形字体+阿里巴巴iconfont收藏的几百个图形字体,享受图形字体带来的乐趣。
  13. 所有控件最后生成一个dll动态库文件,可以直接集成到qtcreator中拖曳设计使用。
  14. 目前已经有qml版本,后期会考虑出pyqt版本,如果用户需求量很大的话。

六、SDK下载

  • SDK下载链接:https://pan.baidu.com/s/1tD9v1YPfE2fgYoK6lqUr1Q 提取码:lyhk
  • 自定义控件+属性设计器欣赏:https://pan.baidu.com/s/1l6L3rKSiLu_uYi7lnL3ibQ 提取码:tmvl
  • 下载链接中包含了各个版本的动态库文件,所有控件的头文件,使用demo。
  • 自定义控件插件开放动态库dll使用(永久免费),无任何后门和限制,请放心使用。
  • 目前已提供26个版本的dll,其中包括了qt5.12.3 msvc2017 32+64 mingw 32+64 的。
  • 不定期增加控件和完善控件,不定期更新SDK,欢迎各位提出建议,谢谢!
  • widget版本(QQ:517216493)qml版本(QQ:373955953)三峰驼(QQ:278969898)。
  • 涛哥的知乎专栏 Qt进阶之路 https://zhuanlan.zhihu.com/TaoQt
  • 欢迎关注微信公众号【高效程序员】,C++/Python、学习方法、写作技巧、热门技术、职场发展等内容,干货多多,福利多多!

Qt编写自定义控件25-自定义QCustomPlot的更多相关文章

  1. Qt编写自定义控件二动画按钮

    现在的web发展越来越快,很多流行的布局样式,都是从web开始的,写惯了Qt widgets 项目,很多时候想改进一下现有的人机交互,尤其是在现有的按钮上加一些动画的效果,例如鼠标移上去变大,移开还原 ...

  2. Qt编写自定义控件38-高亮按钮

    一.前言 高亮按钮控件,既可以作为类似于交通指示灯使用,也可以作为设备状态指示灯使用,控件内置多套颜色风格,还可以自己设置颜色风格,按钮可以增加文字显示,非常适合需要在状态设备上显示小量的文字展示,按 ...

  3. Qt编写自定义控件24-图片轮播控件

    一.前言 上一篇文章写的广告轮播控件,采用的传统widget堆积设置样式表做的,这次必须要用到更高级的QPainter来绘制了,这个才是最高效的办法,本控件参考雨田哥的轮播控件,经过大规模的改造而成, ...

  4. Qt编写自定义控件20-自定义饼图

    前言 上次在写可视化数据大屏电子看板项目的时候,为了逐步移除对QChart的依赖(主要是因为QChart真的太垃圾了,是所有Qt的模块中源码最烂的一个,看过源码的人没有一个不吐槽,不仅不支持10W级别 ...

  5. Qt编写自定义控件9-导航按钮控件

    前言 导航按钮控件,主要用于各种漂亮精美的导航条,我们经常在web中看到导航条都非常精美,都是html+css+js实现的,还自带动画过度效果,Qt提供的qss其实也是无敌的,支持基本上所有的CSS2 ...

  6. Qt编写自定义控件7-自定义可拖动多边形

    前言 自定义可拖动多边形控件,原创作者是赵彦博(QQ:408815041 zyb920@hotmail.com),创作之初主要是为了能够在视频区域内用户自定义可拖动的多个区域,即可用来作为警戒区域,也 ...

  7. Qt编写自定义控件12-进度仪表盘

    前言 进度仪表盘主要应用场景是标识一个任务进度完成的状况等,可以自由的设置范围值和当前值,为了美观还提供了四种指示器(圆形指示器/指针指示器/圆角指针指示器/三角形指示器),各种颜色都可以设置,其中的 ...

  8. Qt编写自定义控件72-提示进度条

    一.前言 我们在很多的安装包中,在安装过程中,经常可以在底部看到一个漂亮的进度条,上面悬浮着显示对应的进度,然后底部进度多种颜色渐变展示,Qt自带的进度条或者操作系统的进度条样式,不够炫,这次索性直接 ...

  9. Qt编写自定义控件71-圆弧进度条

    一.前言 现在web形式的图表框架非常流行,国产代表就是echart,本人用过几次,三个字屌爆了来形容,非常强大,而且易用性也非常棒,还是开源免费的,使用起来不要太爽,内置的各种图表和仪表盘等非常丰富 ...

随机推荐

  1. Linux学习笔记(十四)磁盘管理(二):格式化、挂载以及Swap分区

    一.格式化 第一种写法 mkfs.文件系统 [分区名称(设备文件路径)] 例如:对sdb硬盘的第一个分区以ext3文件系统进行格式化 第二种写法 mkfs -t 文件系统  [分区名称(设备文件路径) ...

  2. ZZNU-OJ-2098 : Drink coffee【线段树合并区间或者 差分 + 二分索引树】

    : Drink coffee 时间限制: Sec 内存限制: MiB 提交: 答案正确: 提交 状态 讨论区 题目描述 为了在上课时保持清醒,凯伦需要一些咖啡.咖啡爱好者凯伦想知道最佳的温度来冲煮完美 ...

  3. hbase实践之flush and compaction

    本文主要涉及flush流程,探讨flush流程过程中引入的问题并阐述2种解决策略,最后简要说明Flush执行策略. 对于Compaction,本文主要探讨Compaction要解决的本质问题以及由Co ...

  4. 6-修改虚拟机的SID(sysprep)

    一.注意点 1.这种错误一般会出现在克隆的机器里,导致在加入域的时候,报SID一致,加入失败,如下图: 2.使用sysprep修改完之后,需要重启服务器,并且重新配置,重新配网络: 3.如果修改SID ...

  5. python numpy的基本操作

    站长资讯平台:文章目录0.NumPy 与 ndarry1.数组属性查看:类型.尺寸.形状.维度2.numpy元素中数据存储方式,数据类型,类型转换2.1 查看元素数据存储类型2.2 元素数据存储类型转 ...

  6. 后端数据中含有html标签和css样式,前端如何转译展示样式效果。

    后端含有html标签和css样式的数据: domain="<span style='color:red'>www.baidu.com</span>" (vu ...

  7. 15、生命周期-BeanPostProcessor-后置处理器

    15.生命周期-BeanPostProcessor-后置处理器 BeanPostProcessor 接口 package org.springframework.beans.factory.confi ...

  8. 怎么学好js

    Js给人那种感觉的原因多半是因为它如下的特点: A:本身知识很抽象.晦涩难懂,如:闭包.内置对象.DOM. B:本身内容很多,如函数库.对象库就一大堆. C:混合多种编程思想.它里面不但牵涉面向过程编 ...

  9. 【题解】[NOIP模拟题]我要的幸福-C++

    题目Description我要的幸福(happiness)幸福/我要的幸福/渐渐清楚/梦想/理想/幻想/狂想/妄想/我只想坚持每一步/该走的方向/就算一路上/偶尔会沮丧/生活是自己/选择的衣裳/幸福/ ...

  10. 010_Linux驱动之_DECLARE_WAIT_QUEUE_HEAD函数

    1. 功能:创建以个等待队列头 2. 函数原型 #define DECLARE_WAIT_QUEUE_HEAD (name)                            /     wait ...