《QT实用小工具·四十五》可以在界面上游泳的小鱼

devtools/2024/9/25 7:39:35/

1、概述
源码放在文章末尾

该项目实现了灵动的小鱼,可以在界面上跟随鼠标点击自由的游泳,项目demo演示如下所示:
在这里插入图片描述

项目部分代码如下所示:

#include "magicfish.h"
#include <QtMath>
#include <QPainter>
#include <QPainterPath>
#include <QVariantAnimation>MagicFish::MagicFish(QQuickPaintedItem *parent): QQuickPaintedItem(parent),m_fishRadius(30),m_finLen(30 * 1.3),m_bodyHeight(30 * 3.2),m_headAlpha(200),m_bodyAlpha(225),M_finAlpha(120),m_mainAngle(0.0),m_curValue(0),m_wave(1.0),m_startFin(false),m_paintPoint(false)
{m_animation = new QVariantAnimation(this);m_animation->setDuration(180 * 1000);m_animation->setStartValue(0);m_animation->setEndValue(54000);m_animation->setLoopCount(-1);connect(m_animation, &QVariantAnimation::valueChanged, this, [this](const QVariant &value){m_curValue = value.toInt();update();});connect(this, &QQuickItem::widthChanged, this, &MagicFish::resize);connect(this, &QQuickItem::heightChanged, this, &MagicFish::resize);m_animation->start();
}void MagicFish::paint(QPainter *painter)
{painter->setRenderHint(QPainter::Antialiasing);QPointF middle_pos = QPointF(width() / 2, height() / 2);m_headPos = calcPoint(middle_pos, m_bodyHeight / 2.0, m_mainAngle);paintMyPoint(painter, m_headPos);paintMyPoint(painter, middle_pos);painter->setPen(Qt::NoPen);painter->setBrush(QBrush(QColor(20, 203, 232, 50)));painter->setBrush(QBrush(QColor(244, 92, 71, m_headAlpha)));painter->drawEllipse(m_headPos, m_fishRadius, m_fishRadius);qreal angle = m_mainAngle + qSin(qDegreesToRadians(m_curValue * 1.2 * m_wave)) * 2;QPointF end_pos = calcPoint(m_headPos, m_bodyHeight, angle - 180);QPointF pos1 = calcPoint(m_headPos, m_fishRadius, angle - 80);QPointF pos2 = calcPoint(end_pos, m_fishRadius * 0.7, angle - 90);QPointF pos3 = calcPoint(end_pos, m_fishRadius * 0.7, angle + 90);QPointF pos4 = calcPoint(m_headPos, m_fishRadius, angle + 80);QPointF central_left = calcPoint(m_headPos, m_bodyHeight * 0.56, angle - 130);QPointF central_right = calcPoint(m_headPos, m_bodyHeight * 0.56, angle + 130);QPainterPath path;path.moveTo(pos1);path.quadTo(central_left, pos2);path.lineTo(pos3);path.quadTo(central_right, pos4);path.lineTo(pos1);painter->setBrush(QBrush(QColor(244, 92, 71, m_bodyAlpha)));painter->drawPath(path);paintMyBody(painter, end_pos, m_fishRadius * 0.7, 0.6, angle);QPointF left_fin_pos = calcPoint(m_headPos, m_fishRadius * 0.9, angle + 110);paintMyFishFins(painter, left_fin_pos, true, angle);QPointF right_fin_pos = calcPoint(m_headPos, m_fishRadius * 0.9, angle - 110);paintMyFishFins(painter, right_fin_pos, false, angle);
}void MagicFish::resize()
{m_fishRadius = qMin(width(), height()) / 10.0;m_finLen = m_fishRadius * 1.3;m_bodyHeight = m_fishRadius * 3.2;
}QPointF MagicFish::calcPoint(const QPointF &pos, qreal length, qreal angle)
{qreal delta_x = qCos(qDegreesToRadians(angle)) * length;qreal delta_y = qSin(qDegreesToRadians(angle - 180)) * length;return QPointF(pos + QPointF(delta_x, delta_y));
}void MagicFish::paintMyPoint(QPainter *painter, const QPointF pos)
{if(m_paintPoint){painter->save();painter->setPen(QPen(Qt::black, 3));painter->setBrush(QBrush(Qt::black));painter->drawPoint(pos);painter->restore();}
}void MagicFish::paintMyFishFins(QPainter *painter, const QPointF &pos, bool is_left, qreal father_angle)
{qreal contral_angle = 115;qreal fin_angle = m_startFin ? qSin(qDegreesToRadians(m_curValue * 16.1 * m_wave)) * 12.0 : 2;QPainterPath path;path.moveTo(pos);QPointF end_pos = calcPoint(pos, m_finLen, is_left ? father_angle + fin_angle + 180 :father_angle - fin_angle - 180);QPointF control_pos = calcPoint(pos, m_finLen * 1.8, is_left ?father_angle + contral_angle + fin_angle :father_angle - contral_angle - fin_angle);path.quadTo(control_pos, end_pos);path.lineTo(pos);painter->save();painter->setBrush(QBrush(QColor(244, 92, 71, M_finAlpha)));painter->drawPath(path);painter->restore();
}void MagicFish::paintMyBody(QPainter *painter, const QPointF &pos, qreal seg_r, qreal MP, qreal father_angle)
{qreal angle = father_angle + qCos(qDegreesToRadians(m_curValue * 1.5 * m_wave)) * 15;qreal length = seg_r * (MP + 1);QPointF end_pos = calcPoint(pos, length, angle - 180);QPointF pos1 = calcPoint(pos, seg_r, angle - 90);QPointF pos2 = calcPoint(end_pos, seg_r * MP, angle - 90);QPointF pos3 = calcPoint(end_pos, seg_r * MP, angle + 90);QPointF pos4 = calcPoint(pos, seg_r, angle + 90);painter->save();painter->setBrush(QBrush(QColor(244, 92, 71, m_headAlpha)));painter->drawEllipse(pos, seg_r, seg_r);painter->drawEllipse(end_pos, seg_r * MP, seg_r * MP);QPainterPath path;path.moveTo(pos1);path.lineTo(pos2);path.lineTo(pos3);path.lineTo(pos4);painter->drawPath(path);painter->restore();paintMyBody2(painter, end_pos, seg_r * 0.6, 0.4, angle);
}void MagicFish::paintMyBody2(QPainter *painter, const QPointF &pos, qreal seg_r, qreal MP, qreal father_angle)
{qreal angle = father_angle + qSin(qDegreesToRadians(m_curValue * 1.5 * m_wave)) * 35;qreal length = seg_r * (MP + 2.7);QPointF end_pos = calcPoint(pos, length, angle - 180);QPointF pos1 = calcPoint(pos, seg_r, angle - 90);QPointF pos2 = calcPoint(end_pos, seg_r * MP, angle - 90);QPointF pos3 = calcPoint(end_pos, seg_r * MP, angle + 90);QPointF pos4 = calcPoint(pos, seg_r, angle + 90);paintMyTail(painter, pos, length, seg_r, angle);painter->save();painter->setBrush(QBrush(QColor(244, 92, 71, m_headAlpha)));painter->drawEllipse(end_pos, seg_r * MP, seg_r * MP);QPainterPath path;path.moveTo(pos1);path.lineTo(pos2);path.lineTo(pos3);path.lineTo(pos4);painter->drawPath(path);painter->restore();
}void MagicFish::paintMyTail(QPainter *painter, const QPointF &pos, qreal length, qreal max_w, qreal angle)
{qreal w = qAbs(qSin(qDegreesToRadians(m_curValue * 1.7 * m_wave)) * max_w + m_fishRadius / 5.0 * 3.0);QPointF end_point1 = calcPoint(pos, length, angle - 180);QPointF end_point2 = calcPoint(pos, length - 10, angle - 180);QPointF pos1 = calcPoint(end_point1, w, angle - 90);QPointF pos2 = calcPoint(end_point1, w, angle + 90);QPointF pos3 = calcPoint(end_point2, w - m_fishRadius / 1.5, angle - 90);QPointF pos4 = calcPoint(end_point2, w - m_fishRadius / 1.5, angle + 90);QPainterPath path;path.moveTo(pos);path.lineTo(pos3);path.lineTo(pos4);path.lineTo(pos);painter->save();painter->setBrush(QBrush(QColor(244, 92, 71,     m_headAlpha)));painter->drawPath(path);path.closeSubpath();path.moveTo(pos);path.lineTo(pos1);path.lineTo(pos2);path.lineTo(pos);painter->drawPath(path);painter->restore();
}void MagicFish::setWave(qreal value)
{m_wave = value;
}qreal MagicFish::getFishR() const
{return m_fishRadius;
}qreal MagicFish::getAngle()
{return m_mainAngle;
}QRectF MagicFish::geometry() const
{return QRectF(x(), y(), width(), height());
}QPointF MagicFish::getHeadPos() const
{return m_headPos;
}void MagicFish::setCurrentAngle(qreal angle)
{m_mainAngle = angle;update();
}void MagicFish::setFinAnimation(bool start)
{m_startFin = start;
}void MagicFish::setFishR(int value)
{m_fishRadius = value;m_finLen = value * 1.3;m_bodyHeight = value * 3.2;update();
}

源码下载


http://www.ppmy.cn/devtools/22968.html

相关文章

【Vue+ElementUI】el-table动态高度设置及表格内容错乱对不齐

在Vue中使用ElementUI的el-table标签会遇到以下问题 一、遇到页面切换时&#xff0c;表格内容错乱&#xff08;对不齐&#xff09;&#xff1a;doLayout() 二、动态计算表格高度&#xff1a;getTableMaxHeight() 页面结构&#xff1a;搜索框表格&#xff0c;通常会在表格内部…

3分钟了解拍摄VR全景需要哪些硬件

VR全景图片是一张水平方向360度&#xff0c;垂直方向180度&#xff0c; 图片尺寸宽高比为2:1的图片。 通过720yun APP或720yun官网上传生成全景H5页面&#xff0c;即可360度全方位观看画面中的景象。 拍摄VR全景有很多方法&#xff0c;下面介绍用单反相机、全景相机、智能手机…

Linux tcp/ip 网路协议栈学习-00 前言

Linux tcp/ip 网路协议栈学习-00 前言 目录 Linux tcp/ip 网路协议栈学习-00 前言 (1)预备知识 (2)前置知识 (3)学习目标 (4)总结 (1)预备知识 好工具事半功倍&#xff0c;做任何事情都需要有方法和工具&#xff0c;同样&#xff0c;阅读 Linux 内核源码也是如此…

python代码实现支持向量机对鸢尾花分类

1、导入支持向量机模型&#xff0c;划分数据集 from sklearn import datasets from sklearn import svmirisdatasets.load_iris() iris_xiris.data iris_yiris.target indices np.random.permutation(len(iris_x)) iris_x_train iris_x[indices[:-10]] iris_y_train iris_y…

HTTP:强缓存优化实践

强缓存&#xff1a;浏览器不会向服务器发送任何请求&#xff0c;直接从本地缓存中读取文件 强缓存是指浏览器在向服务器请求资源时&#xff0c;判断本地是否存在该资源的缓存&#xff0c;并判断是否过期。 如果本地缓存未过期&#xff0c;浏览器就直接使用本地缓存&#xff0c…

正点原子[第二期]Linux之ARM(MX6U)裸机篇学习笔记-6.5

前言&#xff1a; 本文是根据哔哩哔哩网站上“正点原子[第二期]Linux之ARM&#xff08;MX6U&#xff09;裸机篇”视频的学习笔记&#xff0c;在这里会记录下正点原子 I.MX6ULL 开发板的配套视频教程所作的实验和学习笔记内容。本文大量引用了正点原子教学视频和链接中的内容。…

软考-论文写作-论架构风格论文

题目 素材 框架 一、 摘要 2020年12月,我参加了某省政协委员履职系统的开发。该系统为政协机关人员线上开展各项工作以及委员完成各项履职提供了全方位的软件支撑。我在该项目重担任系统架构师一职,负责履职系统的架构设计。本文结合实践,以委员履职系统为例,主要讨论软件…

在VSCode中配置多个版本的Python环境,并设置PYTHONHOME环境变量

在VSCode中配置多个版本的Python环境&#xff0c;并设置PYTHONHOME环境变量&#xff0c;可以通过以下步骤进行&#xff1a; 安装多个版本的Python 首先&#xff0c;你需要在你的计算机上安装多个版本的Python。你可以从Python的官方网站下载不同版本的Python安装包&#xff0…