Qt QWiget 实现简约美观的加载动画 第三季

😃 第三季来啦 😃 这是最终效果:
在这里插入图片描述
只有三个文件,可以直接编译运行

//main.cpp
#include "LoadingAnimWidget.h"
#include <QApplication>
#include <QVBoxLayout>
#include <QGridLayout>
int main(int argc, char *argv[])
{QApplication a(argc, argv);QWidget w;w.setWindowTitle("加载动画 第三季");QGridLayout * mainLayout = new QGridLayout;auto* anim1= new RippleWave;mainLayout->addWidget(anim1,0,0);auto* anim2 = new SlinkyCircle;mainLayout->addWidget(anim2,0,1);auto * anim3 = new ZoomingBalls;mainLayout->addWidget(anim3,1,0);auto* anim4 = new SpotFoldCircle;mainLayout->addWidget(anim4,1,1);w.setLayout(mainLayout);w.show();anim1->start();anim2->start();anim3->start();anim4->start();return a.exec();
}
//LoadingAnimWidget.cpp
#include "LoadingAnimWidget.h"
#include <QDebug>
#include <QPaintEvent>
#include <QPainter>
LoadingAnimBase::LoadingAnimBase(QWidget* parent):QWidget(parent){mAnim.setPropertyName("angle");mAnim.setTargetObject(this);mAnim.setDuration(2000);mAnim.setLoopCount(-1);//run forevermAnim.setEasingCurve(QEasingCurve::Linear);setFixedSize(200,200);mAngle = 0;
}
LoadingAnimBase::~LoadingAnimBase(){}
void LoadingAnimBase::exec(){if(mAnim.state() == QAbstractAnimation::Stopped){start();}else{stop();}
}
void LoadingAnimBase::start(){mAnim.setStartValue(0);mAnim.setEndValue(360);mAnim.start();
}
void LoadingAnimBase::stop(){mAnim.stop();
}
qreal LoadingAnimBase::angle()const{ return mAngle;}
void LoadingAnimBase::setAngle(qreal an){mAngle = an;update();
}SlinkyCircle::SlinkyCircle(QWidget* parent):LoadingAnimBase (parent){mAnim.setEasingCurve(QEasingCurve::InOutCubic);
}
void SlinkyCircle::paintEvent(QPaintEvent *event){QPainter painter(this);painter.setRenderHint(QPainter::Antialiasing);painter.setPen(Qt::NoPen);//画圆弧painter.setBrush(Qt::NoBrush);const int x = this->width();const int y = this->height();QPen pen("black");pen.setCapStyle(Qt::RoundCap);pen.setWidth(x/20);painter.setPen(pen);painter.translate(x/2,y/2);static const qreal spanAngle = 90;//mAngle<=45,要把弧线拉伸出来static const qreal shrinkAngle = 360 - spanAngle;//mAngle==315时,要把弧线收缩起来auto arcRect = this->rect().adjusted(x/5,y/5,-x/5,-y/5);arcRect.translate(-x/2,-y/2);static const int direction = -1;//顺时针if(mAngle < spanAngle){painter.drawArc(arcRect,90 * 16,mAngle * 16 * direction);}else{//弧长是固定的//40 - 320 --> 320 , 280 --> 320if(mAngle > shrinkAngle){painter.drawArc(arcRect,90 * 16,-(360-mAngle)*16 * direction);}else{//我转动的角度是当前角度 - 拉伸门槛,因为有收尾的不动的时间段,占据了一段角度,所以要把转动的角度拉伸一些,//这个比例就是 (360-spanAngle) / (shrinkAngle - spanAngle)const auto delta = (mAngle - spanAngle) * (360-spanAngle) / (shrinkAngle - spanAngle);painter.rotate(-delta * direction);painter.drawArc(arcRect,90 * 16,spanAngle * 16 * direction);}}}
ZoomingBalls::ZoomingBalls(QWidget* parent):LoadingAnimBase (parent){}void ZoomingBalls::paintEvent(QPaintEvent *event){QPainter painter(this);painter.setRenderHint(QPainter::Antialiasing);painter.setPen(Qt::NoPen);//这里可以设置小球的颜色,也可以改成类的成员static const QStringList colorList{"black","black","black","black","black"};const int x = width();const int y = height();const qreal r = x/24;    //小球的半径const qreal rmax = x/10;  //小球最大半径const qreal alphaSpace = 0.8; //小球颜色变化范围static const int startAngle[5] = {0,45,90,135,180};//五个小球开始变化时间点painter.translate(x/6,y/2);for(int i = 0;i < colorList.size();++i){const auto start = startAngle[i];qreal delta = mAngle - start;QColor background = colorList[i];if(delta > 0 && delta < 180){if(delta > 90) delta = 180 - delta;qreal ratio = delta/ 90.0;qreal alpha = 1- ratio *alphaSpace;qreal radius = r + ratio * (rmax - r);background.setAlphaF(alpha);painter.setBrush(QBrush(background));painter.drawEllipse(QPointF(0,0),radius,radius);}else{//不变painter.setBrush(QBrush(background));painter.drawEllipse(QPointF(0,0),r,r);}painter.translate(x/6.0,0);}
}
SpotFoldCircle::SpotFoldCircle(QWidget* parent):LoadingAnimBase (parent){}void SpotFoldCircle::paintEvent(QPaintEvent *event){QPainter painter(this);painter.setRenderHint(QPainter::Antialiasing);painter.setPen(Qt::NoPen);painter.setBrush(QBrush(QColor("black")));static const int startAngle[5] = {0,15,28,40,50}; //小球开始转动的开始时间点static const int restAngle = 360 - startAngle[4]; //小球休息的时间点static const qreal angleStretch = 360.0 / (360 - startAngle[4]); //角度拉伸比例const int x = this->width();const int y = this->height();painter.translate(x/2,y/2);const int radius = x / 16;//小球的半径static const qreal ratioList[5] = {1,0.8,0.64,0.512,0.41}; //小球半径比例列表 ,很明显,小球越来越小for(int i = 0;i < 5;++i){const int start = startAngle[i];const qreal r = radius * ratioList[i];qreal delta = mAngle - start;if(delta > 0 && delta < restAngle){ //要转动起来painter.rotate(delta * angleStretch);painter.drawEllipse(QPointF(0,-y/2 + radius*2),r,r);painter.rotate(-delta * angleStretch);}else{ //停在原地painter.drawEllipse(QPointF(0,-y/2 + radius*2),r,r);}}
}
RippleWave::RippleWave(QWidget* parent):LoadingAnimBase (parent){}void RippleWave::paintEvent(QPaintEvent *event){QPainter painter(this);painter.setRenderHint(QPainter::Antialiasing);painter.setBrush(Qt::NoBrush);QPen pen("black");pen.setWidth(3);//每层波浪的生命周期是6个单位,一共有八层,每层开始时间是前一层波浪的一个单位之后,时间共分为14个单位static const int arr[8] = {10,15,22,31,42,55,70,87};//波浪的半径static const qreal unit = 360.0 / 14;static const qreal startArr[8] = {0,unit,unit*2 , unit*3 ,unit*4,unit*5,unit*6,unit*7};static const qreal lifeSpan = unit*6;const int x = this->width();const int y = this->height();painter.translate(x/2,y/2);for(int i = 0;i < sizeof(arr) / sizeof(int);++i){const auto start = startArr[i];const auto delta = mAngle - start;qreal alpha = 1;if(delta > 0 && delta < lifeSpan){if(delta < unit*2) alpha = delta / (unit*2);if(delta > unit*4) alpha = (lifeSpan - delta) / (unit*2);QColor c("black");c.setAlphaF(alpha);pen.setColor(c);painter.setPen(pen);painter.drawEllipse(QPointF(0,0),arr[i],arr[i]);}else{//什么都不用做}}
}
//LoadingAnimWidget.h
#ifndef LOADINGANIMWIDGET_H
#define LOADINGANIMWIDGET_H
#include <QPropertyAnimation>
#include <QWidget>
class LoadingAnimBase:public QWidget
{Q_OBJECTQ_PROPERTY(qreal angle READ angle WRITE setAngle)
public:LoadingAnimBase(QWidget* parent=nullptr);virtual ~LoadingAnimBase();qreal angle()const;void setAngle(qreal an);
public slots:virtual void exec();virtual void start();virtual void stop();
protected:QPropertyAnimation mAnim;qreal mAngle;
};
class SlinkyCircle:public LoadingAnimBase{//可以伸缩的管子,绕着中心转动,就像弹簧圈,英语叫做slinky,查了好久才查到这个单词,有点强迫症😂Q_OBJECT
public:explicit SlinkyCircle(QWidget* parent = nullptr);
protected:void paintEvent(QPaintEvent *event);
};class ZoomingBalls:public LoadingAnimBase{//五个小球,每个依次变大Q_OBJECT
public:explicit ZoomingBalls(QWidget* parent = nullptr);
protected:void paintEvent(QPaintEvent *event);
};class SpotFoldCircle:public LoadingAnimBase{//五个小球绕中心旋转,可以折叠在一起Q_OBJECT
public:explicit SpotFoldCircle(QWidget* parent = nullptr);
protected:void paintEvent(QPaintEvent *event);
};class RippleWave:public LoadingAnimBase{//8层波纹,从中间逐渐向外变亮,然后变暗Q_OBJECT
public:explicit RippleWave(QWidget* parent = nullptr);
protected:void paintEvent(QPaintEvent *event);
};#endif // LOADINGANIMWIDGET_H

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/703953.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

设计并实现一个并发安全的LRU(Least Recently Used,最近最少使用)缓存结构

文章目录 前言实战演示写在最后 前言 相信很多人都使用过LinkedHashMap&#xff0c;LinkedHashMap中的removeEldestEntry可以删除老旧的元素&#xff0c;我们可以以此来实现一个LRU缓存结构&#xff0c;并结合java中JUC包中的各种多线程锁机制来保证多线程安全。 以下是我遇见…

过滤器和拦截器的区别是什么

过滤器和拦截器都是Web应用程序中的常用组件&#xff0c;不过它们的功能和使用方式不同。 1.工作位置不同 过滤器在Servlet容器内部工作 拦截器在SpringMVC框架内部工作 2.触发时间不同 过滤器在客户端向服务器发送请求前触发&#xff0c;在请求到达Servlet容器之前进行处理…

突破编程_C++_面试(模板编程(2))

面试题 1 &#xff1a;什么是模板特化&#xff1f;为什么需要它&#xff1f; 模板特化是C编程语言中的一个概念&#xff0c;它允许我们为模板定义特定类型的版本。简单来说&#xff0c;模板特化是模板的具体化&#xff0c;它允许为特定的类型提供特定的实现。 模板特化主要有…

【前端素材】推荐优质后台管理系统Dashmin平台模板(附源码)

一、需求分析 后台管理系统在多个层次上提供了丰富的功能和细致的管理手段&#xff0c;帮助管理员轻松管理和控制系统的各个方面。其灵活性和可扩展性使得后台管理系统成为各种网站、应用程序和系统不可或缺的管理工具。 后台管理系统是一种具有多层次结构的软件系统&#xf…

主从边沿触发寄存器、SRAM、DRAM

主从边沿触发寄存器SRAMDRAM

工厂设计模式总结

一、简单工厂 1.1 概述 背景&#xff1a; 代码中存在根据不同条件创建不同对象的场景。例如&#xff1a; if ("json".equals(name)) {return new JsonConfigParser(); } else if ("xml".equals(name)) {return new XmlConfigParser(); } else if ("…

邀请函 | 2024年数据技术嘉年华集结号已吹响,期待您参会!

龙腾四海内&#xff0c;风云际会时&#xff0c;2024年中国数据嘉年华如约而至。从起初小范围的网友聚会&#xff0c;到如今面向全国各地从业者、爱好者的年度集会&#xff0c;纵使岁月更迭&#xff0c;我们初心依旧。我们在各自最好的年华里共同见证了中国数据库行业的蓬勃发展…

Linux下的IO多路复用

文章目录 一. IO的概念和分类1. IO操作的原理&#xff1a;二. I/O多路复用使用场景和作用1. 问题&#xff1a; 一台网络服务器需要接收100台客户端的连接和数据通信&#xff0c;应该如何设计和实现&#xff1f;2. I/O多路复用机制&#xff1a; 三. Select poll epollselectpoll…

【前端素材】推荐优质后台管理系统Modernize平台模板(附源码)

一、需求分析 后台管理系统是一种用于管理和控制网站、应用程序或系统后台操作的软件工具&#xff0c;通常由授权用户&#xff08;如管理员、编辑人员等&#xff09;使用。它提供了一种用户友好的方式来管理网站或应用程序的内容、用户、数据等方面的操作&#xff0c;并且通常…

光学3D表面轮廓仪微纳米三维形貌一键测量

光学3D表面轮廓仪(白光干涉仪)利用白光干涉原理&#xff0c;以0.1nm分辨率精准捕捉物体的表面细节&#xff0c;实现三维显微成像测量&#xff0c;被广泛应用于材料学领域的研究和应用。 了解工作原理与技术 材料学领域中的光学3D表面轮廓仪&#xff0c;也被称为白光干涉仪&am…

Mybatis框架相关问题(面试)

目录 1、什么是Mybatis? 2、Mybatis的优点 3、Mybatis的缺点 4、Mybatis框架的适用场合 5、#{}和${}的区别是什么 {}是预编译处理&#xff0c;${}是字符串替换。 6、实体类中的属性名和表中的字段名不一样&#xff0c;怎么办&#xff1f; 8、编写模糊查询like语句 9、…

【LV14 day9内核模块编译】

模块三要素&#xff1a; 入口函数 出口函数 module_license 一、内核模块基础代码解析 Linux内核的插件机制——内核模块 类似于浏览器、eclipse这些软件的插件开发&#xff0c;Linux提供了一种可以向正在运行的内核中插入新的代码段、在代码段不需要继续运行时也可以从内核…

常用的集合有哪些

常用的三大类集合:Set、List、Map。 1 Set 1) Set 集合属于单列集合,不允许包含重复元素; 2) 判断元素是否重复的标准为对象的 equals 方法,存在时返回 false,不存在返回 true; 3) 元素的排序规则,由相应的实现类决定,分为无序、元素大小排序、写入顺序排序; 4) 初始化…

SpringBoot -【SmartInitializingSingleton】基础使用及应用场景

SmartInitializingSingleton 在继续深入探讨 SmartInitializingSingleton接口之前&#xff0c;让我们先了解一下 Spring Framework 的基本概念和背景。Spring Framework 是一个开源的 JavaEE&#xff08;Java Enterprise Edition&#xff09;全栈&#xff08;full-stack&#x…

力扣 724. 寻找数组的中心下标

思路&#xff1a; 创建两个变量sum和sum1&#xff0c;sum代表左边元素的和&#xff0c;sum1代表右边元素的和 然后假设从数组下标0开始&#xff0c;一直到最后一个作为中心下标 如果sumsum1&#xff0c;返回此时的中心下标 如果所有下标循环完了&#xff0c;发现没有return…

Apipost 数据模型功能API数据重复利用起来

在Apipost数据模型中用户可以预先创建多个数据模型&#xff0c;并在API设计过程中重复利用这些模型来构建API 创建数据模型 在左侧导航点击「数据模型」-「新建数据模型」在右侧工作台配置数据模型参数 引入数据模型 在API设计预定义响应期望下点击引用数据模型&#xff0c;…

Docker安装与基础知识

目录 -----------------Docker 概述--------------------------- 容器化越来越受欢迎&#xff0c;因为容器是&#xff1a; Docker与虚拟机的区别&#xff1a; Docker核心概念&#xff1a; ●镜像 ●容器 ●仓库 -----------------安装 Docker--------------------------…

操作系统——处理机调度

文章目录 进程调度0.概念1.调度分类高级调度低级调度中级调度七状态模型调度对比 2.进程调度进程调度的时机进程调度的方式进程的切换方式调度器/调度程序闲逛进程 3. 调度算法的评价指标CPU利用率系统吞吐量周转时间等待时间响应时间 4. 调度算法先来先服务(FCFS)短作业优先(S…

django rest framework 学习笔记-实战商城3

01用户模块模型定义_哔哩哔哩_bilibili 本博客借鉴至大佬的视频学习笔记 用户模块及商品数据表结构设计 from ckeditor.fields import RichTextField # pip install django-ckeditor from django.db import models# Create your models here. # from wx.richtext import Ri…

MATLAB环境下基于PSO-DBN-ELM方法的图像分类

在纯数据驱动的图像识别方法中&#xff0c;深度信念网络DBN识别模型具备较好的识别性能。对于DBN模型而言&#xff0c;可利用的数据越多&#xff0c;挖掘的信息也越多&#xff0c;建立的模型就越准确。然而DBN本身仍存在一定的不足之处&#xff0c;一方面由于DBN内部包含多层限…