如何在Qt中绘制一个带有动画的弧形进度条?

article/2025/8/23 2:38:25

如何在Qt中绘制一个弧形的进度条

在图形用户界面开发中,进度指示控件(Progress Widget)是非常常见且实用的组件。CCArcProgressWidget 是一个继承自 QWidget 的自定义控件,用于绘制圆弧形进度条。当然,笔者看了眼公开的实现,基本上都非常完善了,笔者在这里添加了一个更好的动画。

在这里插入图片描述

类定义概览

CCArcProgressWidget 类定义在 CCArcProgressWidget.h 中,使用 Qt 元对象系统,通过 Q_OBJECT 宏启用信号与属性机制。该控件支持如下属性绑定:

  • value:当前进度值
  • maxValue:最大进度值
  • displayValue:动画过程中的显示值(与实际 value 异步)

这些属性可被 QML 或动画机制绑定,便于动态效果的呈现。(笔者这里使用的是Q_PROPERTY属性系统公开的,所以QML可用(笑))

静态常量定义(这部分是笔者认为编译的时候可以指定的)

类中定义了多个静态常量,用于控制组件的外观与行为:

  • DURATION:动画持续时间(单位:毫秒),默认为 500ms
  • ARC_WIDTH:圆弧的线宽,默认为 50 像素
  • DEFAULT_VALUE:默认初始值,为 0
  • DEFAULT_MAX:默认最大值,为 100
  • DEF_TEXT_COLOR:默认文本颜色
  • DEF_BKCOLOR:默认背景弧颜色(未完成部分)
  • DEF_ARC_COLOR:默认进度弧颜色(已完成部分)

这些常量使得控件具有清晰的默认状态,便于使用和维护。

属性访问与设置接口

该类提供了一系列 inline 内联函数和公开接口,用于读取与设置进度值及外观样式:

  • int value() const:获取当前进度值
  • void setValue(int val):设置当前进度值(含动画)
  • int maxValue() const:获取最大值
  • void setMaxValue(int max):设置最大值
  • QColor progressArcColor() const / void setProgressArcColor(const QColor&):读取与设置进度弧颜色
  • QColor progressBackgroundColor() const / void setProgressBackgroundColor(const QColor&):读取与设置背景弧颜色
  • QColor progressTextColor() const / void setProgressTextColor(const QColor&):读取与设置文本颜色

所有设置函数内部均会判断是否真正发生变化,避免无谓的刷新,若发生更改则调用 update() 触发重绘。

信号机制

该控件定义了三个信号:

  • valueChanged(int):当用户设置新进度值时发出
  • maxValueChanged(int):当最大值被重新设置时发出
  • displayValueChanged(int):当动画中显示的值发生变化时发出

这些信号便于其他模块(如界面展示、数据记录)实时响应进度的变化。

绘制函数与动画支持

该类重载了 paintEvent 事件处理函数,实现核心绘制逻辑。绘制内容包括三部分:

  • 背景弧:通过 drawBackgroundArc() 绘制未完成部分
  • 进度弧:通过 drawProgressArc() 根据当前动画角度绘制完成部分
  • 中心文本:通过 drawText() 绘制当前数值或状态文字

同时,setupAnimation() 函数用于构建 QPropertyAnimation 动画,使 valuedisplayValue 之间具备平滑过渡效果。动画期间实际值不变,仅 displayValue 动态变化,从而提升用户体验。

私有成员变量

类中使用了如下私有成员保存状态:

  • progress_value:当前进度值
  • progress_display_value:当前显示值(用于动画)
  • progress_max_value:最大进度值
  • progress_minAngleprogress_startAngle:控制弧线的起始与方向(默认从顶部顺时针)
  • progress_arc_colorprogress_backgroundColorprogress_textColor:颜色配置
  • QPropertyAnimation* animation:动画对象指针

这些成员变量共同构成了进度显示的完整状态。

使用示例(简要)

CCArcProgressWidget* widget = new CCArcProgressWidget(this);
widget->setValue(70);
widget->setMaxValue(100);
widget->setProgressArcColor(Qt::blue);
widget->setProgressBackgroundColor(Qt::lightGray);
widget->setProgressTextColor(Qt::black);

以上代码将在界面中创建一个蓝色的圆形进度条,表示当前进度为 70%。

一些实现的细节说明

​ 下面的部分是属性设置的接口,没什么有趣的。

#include "CCArcProgressWidget.h"
#include <QPropertyAnimation>CCArcProgressWidget::CCArcProgressWidget(QWidget* parent): QWidget { parent } {setupAnimation();
}void CCArcProgressWidget::setupAnimation() {animation = new QPropertyAnimation(this, "displayValue");animation->setDuration(DURATION);animation->setEasingCurve(QEasingCurve::OutCubic);
}void CCArcProgressWidget::setValue(int val) {val = qBound(0, val, progress_max_value);if (val == progress_value) // avoid duplicate animationsreturn;progress_value = val;animation->stop();animation->setStartValue(progress_display_value);animation->setEndValue(progress_value);animation->start();
}

​ 下面说下我们的绘制,这里是每一次触发重绘的时候我们的设备实际上进行的绘制。

void CCArcProgressWidget::paintEvent(QPaintEvent* event [[maybe_unused]]) {QPainter painter(this);painter.setRenderHint(QPainter::Antialiasing, true);QRectF baseRect = rect();double side = qMin(baseRect.width(), baseRect.height());// 到这里,是为了获取绘制成正方形而不是椭圆形(不然太难看了)QRectF squareRect((baseRect.width() - side) / 2.0,(baseRect.height() - side) / 2.0,side, side);int margin = ARC_WIDTH + 5;QRectF arcRect = squareRect.adjusted(margin, margin, -margin, -margin);double radius = qMin(arcRect.width(), arcRect.height()) / 2;QPointF center = arcRect.center();double angle = 360.0 * progress_display_value / progress_max_value;angle = qMax<double>(progress_minAngle, -angle);drawBackgroundArc(painter, arcRect);drawProgressArc(painter, arcRect, angle);drawText(painter, center, radius);
}
  1. paintEvent 事件首先确定绘制区域 arcRect,再根据当前 displayValue 计算对应的角度 angle。之后,依次调用:
  • drawBackgroundArc:用圆弧绘制背景轨迹。
  • drawProgressArc:绘制当前进度的圆弧,同时在圆弧末端绘制小圆点,增强视觉效果。
  • drawText:居中绘制当前进度的百分比文本。
void CCArcProgressWidget::drawBackgroundArc(QPainter& painter, const QRectF& arcRect) {QPen pen(progress_backgroundColor, ARC_WIDTH);pen.setCapStyle(Qt::RoundCap);painter.setPen(pen);painter.drawArc(arcRect, progress_startAngle * 16, 360 * 16);
}void CCArcProgressWidget::drawProgressArc(QPainter& painter, const QRectF& arcRect, double angle) {if (angle == 0)return;QConicalGradient gradient(arcRect.center(), progress_startAngle);gradient.setColorAt(0, progress_arc_color.lighter(150));gradient.setColorAt(0.5, progress_arc_color);gradient.setColorAt(1, progress_arc_color.darker(150));QPen pen(QBrush(gradient), ARC_WIDTH);pen.setCapStyle(Qt::FlatCap);painter.setPen(pen);painter.drawArc(arcRect, progress_startAngle * 16, -angle * 16);double spanAngleRad = qDegreesToRadians(progress_startAngle - angle);double cx = arcRect.center().x();double cy = arcRect.center().y();double rx = arcRect.width() / 2;double ry = arcRect.height() / 2;double ex = cx + rx * qCos(spanAngleRad);double ey = cy - ry * qSin(spanAngleRad);QBrush brush(gradient);painter.setBrush(brush);painter.setPen(Qt::NoPen);painter.drawEllipse(QPointF(ex, ey), ARC_WIDTH / 2.0, ARC_WIDTH / 2.0);
}void CCArcProgressWidget::drawText(QPainter& painter, const QPointF& center, double radius) {painter.setFont(QFont("Arial", radius * 0.3, QFont::Bold));painter.setPen(progress_textColor);QString text = QString("%1%").arg(qRound(100.0 * progress_display_value / progress_max_value));QRectF textRect(center.x() - radius, center.y() - radius,radius * 2, radius * 2);painter.drawText(textRect, Qt::AlignCenter, text);
}

http://www.hkcw.cn/article/FFjbymZTjd.shtml

相关文章

体育平台数据服务解决方案:从痛点到落地的全栈技术支持

一、体育平台开发的数据痛点解析 在体育平台开发领域&#xff0c;数据层建设往往成为技术团队的核心挑战&#xff1a; 实时数据获取难自建实时数据采集系统需解决赛事方反爬机制、多源数据同步&#xff08;如比分 / 球员位置 / 赔率&#xff09;、毫秒级延迟控制等问题&#…

去寺庙减脂的人胖了11斤 素食诱惑难抵挡

去寺庙减脂的人胖了11斤!一群原本希望通过吃减脂餐修身养性的打工人和中年人,在体验后第一批人竟胖了11斤,这反转实在惊人。当下生活节奏飞快,很多人被亚健康困扰,身体各种不适。于是大家开始注重养生,寺庙因宁静祥和且素食养生概念深入人心,被不少人视为减脂圣地。人们…

printf 输出格式总结(C语言)和C 中的类型提升规则、默认整型提升

类型格式符示例输出说明十进制整数%d12345输出有符号十进制&#xff08;int 类型&#xff09;十进制整数%u12345输出无符号十进制&#xff08;unsigned int&#xff09;十六进制%x3fa2小写十六进制&#xff08;无前缀&#xff09;十六进制%X3FA2大写十六进制&#xff08;无前缀…

Flutter下的一点实践

目录 1、背景2、refena创世纪代码3、localsend里refena的刷新3.1 初始状态3.2 发起设备扫描流程3.3 扫描过程3.3 刷新界面 4.localsend的设备扫描流程4.1 UDP广播设备注册流程4.2 TCP/HTTP设备注册流程4.3 localsend的服务器初始化工作4.4总结 1、背景 在很久以前&#xff0c;…

英国利物浦汽车冲撞人群事件嫌疑人身份公布

英国利物浦汽车冲撞人群事件嫌疑人身份公布 警方通报调查进展英国默西赛德郡警方发言人当地时间5月29日说,日前在利物浦市中心驾车冲撞球迷的男子面临7项指控,将于30日在利物浦地方法院出庭。据英国媒体报道,默西-柴郡地区皇家检察署授权默西赛德郡警方正式对53岁的嫌疑人保…

「动态规划::状压DP」网格图递推 / AcWing 292|327(C++)

目录 概述 相邻行递推 思路 算法过程 优化方案 空间优化 返回值优化 Code 复杂度 相邻两行递推 思路 算法过程 Code 复杂度 特殊优化&#xff1a;编译期计算 总结 概述 如果我们有一张地图&#xff0c;要求是在符合某类条件的前提在地图上放置最优解&#xff…

深圳海关查获超18公斤摇头丸 科技助力精准打击

近日,深圳邮局海关在跨境转运货物的监管过程中查获了一批伪报为“有机蜡溶香草”的包裹,实际上是摇头丸,总重量达18433.3克。目前,该案件已正式移交至海关缉私部门,进行更深入的调查和追溯。事件起因是海关关员对一批申报品名为“有机蜡溶香草”的转运货物进行例行检查时,…

一根网线连接两台电脑组建局域网

用一根网线分别插在网络端口&#xff0c;修改网络IP地址&#xff0c;假如有A和B&#xff0c;设置A:IP地址192.168.1.1,B&#xff1a;192.168.1.2 接着把网络防火墙关闭&#xff0c;步骤如下&#xff1a; 后面接着右键点击我的电脑&#xff0c;选择属性&#xff0c;打开远程控制…

丰巢回应快递柜消失市民取件扑空!

丰巢回应快递柜消失市民取件扑空。5月27日,上海普陀。市民发帖称包裹被存放在丰巢快递柜,过去取件时快递柜却消失不见了。5月29日,该包裹快递员告诉《正在新闻》,昨天在该小区的66号楼快递柜发生同样事件。对于快递柜搬走的原因,他表示不清楚,快递柜管理人员并没有告知他…

树型表查询方法 —— SQL递归

目录 引言&#xff1a; 自链接查询&#xff1a; 递归查询&#xff1a; 编写service接口实现&#xff1a; 引言&#xff1a; 看下图&#xff0c;这是 course_category 课程分类表的结构&#xff1a; 这张表是一个树型结构&#xff0c;通过父结点id将各元素组成一个树。 我…

高校大数据采集平台产品特色

大数据采集平台是专为高校大数据相关专业打造的智能化数据采集教学与实训工具。平台具有以下核心优势&#xff1a;采用可视化图形界面&#xff0c;无需编程基础&#xff0c;通过简单配置即可快速抓取网页中的文本、链接、图片、视频及文档等全类型数据&#xff0c;并自动存储至…

石家庄铁道大学回应书记打人 学生直播见证冲突

5月28日,石家庄铁道大学一名学生在宿舍里被学院书记殴打,还见了血。当时学生正在直播,许多网友目睹了整个过程。当这件事上了热搜后,学校的回应令人气愤,称学院书记没有问题,是学生先动手的。网友们表示不信,质疑书记去学生宿舍是为了让学生打他。事件起因是石家庄铁道大…

PGSQL结合linux cron定期执行vacuum_full_analyze命令

‌VACUUM FULL ANALYZE 详解‌ 一、核心功能 ‌空间回收与重组‌ 完全重写表数据文件&#xff0c;将碎片化的存储空间合并并返还操作系统&#xff08;普通 VACUUM 仅标记空间可重用&#xff09;。彻底清理死元组&#xff08;已删除或更新的旧数据行&#xff09;&#xff0c;解…

吴艳妮摘铜哽咽鞠躬道歉 带伤参赛展现坚韧精神

5月29日,亚洲田径锦标赛女子100米栏决赛中,吴艳妮以13秒07的成绩获得铜牌。赛后,她走路时显得有些一瘸一拐。在接受采访时,吴艳妮哽咽着向大家道歉,表示很感谢现场观众的支持,但没能为中国队拿到冠军感到非常抱歉。她提到自己的伤还没有完全恢复,不想过多解释,但仍坚信…

XCVP1902-2MSEVSVA6865 Xilinx FPGA Versal Premium SoC/ASIC

XCVP1902-2MSEVSVA6865 Versal Premium SoC/ASIC 单片 FPGA&#xff0c;可提供大容量 FPGA 逻辑仿真和原型设计目标。VP1902的逻辑单元数量增加了 2.2 倍&#xff0c;达到 1850 万个。 VP1902 自适应 SoC 提供最大容量和连接能力&#xff0c;具有可随机存取的逻辑密度和 2.4 倍…

TripGenie:畅游济南旅行规划助手:个人工作纪实(二十一)

这次&#xff0c;我新增了一个济南公交线路的展示界面&#xff0c;济南的公交线路多&#xff0c;且经过的站点覆盖范围广&#xff0c;价格实惠&#xff0c;是出行旅游交通工具的不二之选&#xff0c;我基于此现实情况&#xff0c;觉得做一个新的页面全面展示济南交通。 我选择把…

激励电平与频差的微妙平衡:晶振选型不可忽视的细节

在电子设备的设计中&#xff0c;晶振作为提供稳定时钟信号的关键元件&#xff0c;其选型的正确性直接关系到整个系统的性能与稳定性。而在晶振选型过程中&#xff0c;激励电平与频差之间的微妙平衡常常被工程师们所忽视&#xff0c;然而这一细节却可能对电路的正常运行产生深远…

数字人引领政务新风尚:智能设备助力政务服务

在信息技术飞速发展的今天&#xff0c;政府机构不断探索提升服务效率和改善服务质量的新途径。实时交互数字人在政务服务中的应用正成为一大亮点&#xff0c;通过将“数字公务员”植入各种横屏智能设备中&#xff0c;为民众办理业务提供全程辅助。这种创新不仅优化了政务大厅的…

练习小项目9:打字效果文字展示(多段文字循环+删除+光标闪烁)

项目简介&#xff1a; 本文介绍如何用原生JavaScript实现一个简洁的打字效果&#xff0c;支持&#xff1a; 多段文字循环播放 打字完后暂停一会儿 逐字删除&#xff0c;形成打字机动画感 打字光标闪烁效果 项目适合用于首页欢迎语、提示语等动态文本展示&#xff0c;能提…

【从零开始超详细】Linux系统使用docker + docker-compose部署nacos以及SpringBoot+vue项目详细

Linux系统使用dockerdocker-compose部署nacos以及SpringBootvue项目详细文档 本文章Linux发行版为openEuler 22.03 (LTS-SP2), 多数命令与centos一致, 使用centos的小伙伴也可以参考 不知道自己的服务器是什么发行版的小伙伴可以执行如下命令查看: cat /etc/os-release执行结果…