贝塞尔曲线(Bezier)之水波纹的手机充电动画效果(一)

2024-01-13 10:18

本文主要是介绍贝塞尔曲线(Bezier)之水波纹的手机充电动画效果(一),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

博主声明:

转载请在开头附加本文链接及作者信息,并标记为转载。本文由博主 威威喵 原创,请多支持与指教。

本文首发于此   博主:威威喵  |  博客主页:https://blog.csdn.net/smile_running

    博主这几天一直在搞贝塞尔曲线(Bezier)动画的研究,虽然我的数学不太好,但是也勉勉强强能够看懂懂贝塞尔曲线的公式,套用还是很简单的。前几次搞了几个贝塞尔曲线动画效果,感觉那个效果还是非常赞的,今天兴致又来了,于是去搜索了一下 Android 相关的贝塞尔曲线的动画实例,偶然看到一个 Android 充电进度的贝塞尔曲线动画,它的效果图如下:

     看到这个效果呢,我首先是想到用三阶贝塞尔曲线公式来做,于是就屁颠屁颠的开始了,套了三阶贝塞尔曲线的公式,发现效果没出来,卧槽。害我白高兴一场,以为我的数学还是可以的,结果。。。

    我最先的想法是通过点位去计算波形路径,不过最后放弃了。哈哈,喜出望外,结果我发现了一个更简单的做法,用 Path 类下面的一个三阶贝塞尔曲线的封装方法,很简单就实现了波浪的效果,这是我写这个效果时所收获到的意外惊喜,之前还没字母使用过,接下来我们进行分析这个效果的实现,然后再讲解一下 Path 类三阶贝塞尔的简单用法。

    多的就不扯淡了,我们直接开始吧。国际惯例,先来看看最终的实现效果图:

    这个充电进度的动画效果还行吧,上面我搜索到的是一张静态图,我就是依照这那张图的样式做的,可能颜色又一点点缺陷,这个自己再美化美化就好啦。

    来吧,拿到这个效果图,首先就是分析一波。来看一下草图

    看上面那张图,首先我们要把圆绘制到中心点吧,这没什么问题。因为三阶贝塞尔曲线需要 2 个控制点,从图中我们知道 p1 和 p2 就是那条曲线的控制点, 而且上图 p1 p2 p3 p4 四个点获取坐标都很容易。

        //内部pIn0 = new PointF(rippleX - mDefCircleRadius, rippleY);pIn1 = new PointF(rippleX - mRandom.nextInt((int) mDefCircleRadius), rippleY - mRandom.nextInt((int) (mDefCircleRadius / 4)));pIn2 = new PointF(rippleX + mRandom.nextInt((int) mDefCircleRadius), rippleY + mRandom.nextInt((int) (mDefCircleRadius / 4)));pIn3 = new PointF(rippleX + mDefCircleRadius, rippleY);

    因为海浪波纹有两条曲线组成,这两条曲线是交错的,所以我们需要再来 4 个点

        // 外部pExt0 = new PointF(rippleX - mDefCircleRadius, rippleY);pExt1 = new PointF(rippleX - mRandom.nextInt((int) mDefCircleRadius), rippleY + mRandom.nextInt((int) (mDefCircleRadius / 3)));pExt2 = new PointF(rippleX + mRandom.nextInt((int) mDefCircleRadius), rippleY + mRandom.nextInt((int) (mDefCircleRadius / 3)));pExt3 = new PointF(rippleX + mDefCircleRadius, rippleY);

    得到曲线的点之后呢,我们就可以开始用 Path 类的一个方法去形成曲线的路径了,因为波浪是有颜色的,所以需要把 Path 给封闭起来,形成密闭的效果。接着,再来看一张草图

    用 Path 类制作一条曲线,并且我们要把 p0 ~ p5 这几个点给封闭起来,形成海浪的效果。想法是不错,但是你会发现,这个形成的区域已经超出了圆的范围了吧,那样子就非常丑,犹如这个样子:

    圆圈外面多出了两个蓝色部分区域,丑的不行啊。 像这个样子的情况,我最先想到的是 canvas 有没有画剪切区域的,后来找了一下,好像没找到。陷入深思,后来灵机一动,想到我上一次实现的一种效果,是画一个圆,从内到外扩散的,感兴趣的可以点击链接,去看看我的文章:Android 视差动画 — 雅虎新闻内容揭示效果

    这个圆效果呢,就是从小变到大,逐渐的把内容呈现出来。这就给我一个很好的启示,我可以绘制一个这样的圆,把外面蓝色部分遮住不久好了嘛,也就相当于除了绿色包含的圆以外全部给遮住,这样显示的效果只能看到这个绿色的圆了,我们的目的也就达到了。这个就需要对画笔的宽度进行计算,代码如下:

    private void drawMasked(Canvas canvas) {//绘制一个遮罩层,屏蔽 Path Close 以外的区域mMaskPaint.setStrokeWidth(mDiagonal + mDefCircleRadius * 2 - mPaintSize * 1.5f);canvas.drawCircle(mCircleX, mCircleY, mDiagonal, mMaskPaint);}

    这样就把露出来的蓝色区域给遮挡住了,接下来还有一个难点,就是如何根据进度值把海浪也给升高,总不能在固定位置浪啊浪吧。这就要考虑一个问题,我们需要根据圆的直径和进度值的一个比例关系,计算出当前海平面的高度,通过不断的增加 progress(进度),海平面会随着进度升高,而且这个期间波浪一直在流动的。这部分关键代码如下:

        // 直径与进度的比例rippleScale = 2 * mDefCircleRadius / 100;// 绘制海浪的波纹效果,分内部和外部两条private void drawExternalRipple(Canvas canvas) {// 计算进度的 x , y 位置y = mCircleY - mDefCircleRadius + (100 - mProgress) * rippleScale;x = caculateX(y);float rippleY = y;float rippleX = mCircleX;//内部pIn0 = new PointF(rippleX - mDefCircleRadius, rippleY);pIn1 = new PointF(rippleX - mRandom.nextInt((int) mDefCircleRadius), rippleY - mRandom.nextInt((int) (mDefCircleRadius / 4)));pIn2 = new PointF(rippleX + mRandom.nextInt((int) mDefCircleRadius), rippleY + mRandom.nextInt((int) (mDefCircleRadius / 4)));pIn3 = new PointF(rippleX + mDefCircleRadius, rippleY);Path inPath = new Path();inPath.moveTo(pIn0.x, pIn0.y);inPath.cubicTo(pIn1.x, pIn1.y, pIn2.x, pIn2.y, pIn3.x, pIn3.y);inPath.lineTo(mCircleX + mDefCircleRadius, mCircleY + mDefCircleRadius);inPath.lineTo(mCircleX - mDefCircleRadius, mCircleY + mDefCircleRadius);inPath.close();canvas.drawPath(inPath, mInnerPaint);// 外部pExt0 = new PointF(rippleX - mDefCircleRadius, rippleY);pExt1 = new PointF(rippleX - mRandom.nextInt((int) mDefCircleRadius), rippleY + mRandom.nextInt((int) (mDefCircleRadius / 3)));pExt2 = new PointF(rippleX + mRandom.nextInt((int) mDefCircleRadius), rippleY + mRandom.nextInt((int) (mDefCircleRadius / 3)));pExt3 = new PointF(rippleX + mDefCircleRadius, rippleY);Path extPath = new Path();extPath.moveTo(pExt0.x, pExt0.y);extPath.cubicTo(pExt1.x, pExt1.y, pExt2.x, pExt2.y, pExt3.x, pExt3.y);extPath.lineTo(mCircleX + mDefCircleRadius, mCircleY + mDefCircleRadius);extPath.lineTo(mCircleX - mDefCircleRadius, mCircleY + mDefCircleRadius);extPath.close();canvas.drawPath(extPath, mExternalPaint);}

    上面代码是计算进度条和圆的直径的比例,通过这个比例,我们可以拿到 path 中波浪逐渐上升的 y 坐标,通过不断的绘制 path 然后形成波浪的动画效果,直到进度条为 100 时,我们就进行判断处理

    public void setProgress(int progress) {this.mProgress = progress;this.mArcProgress = mProgress * 3.6f;if (mProgress <= 100) {isFinished = false;} else {isFinished = true;}invalidate();}

    如果进度达到 100,我们就开始绘制完成时候的动画,代码如下

    private void drawFinished(Canvas canvas) {canvas.drawCircle(mCircleX, mCircleY, mDefCircleRadius, mArcPaint);canvas.drawCircle(mCircleX, mCircleY, mDefCircleRadius, mInnerPaint);canvas.drawText("充电完成", mCircleX - mTextPaint.getTextSize() * 2f, mCircleY + mTextPaint.getTextSize() / 2, mTextPaint);}

    只有这样,当结束是才会显示不同的效果,否则不做处理的话,就是空空如也啦。

     那么至此,我们对这个效果的分析也就完成了,并且手动进实现了一下,感觉收获了不少,哈哈。最后呢,给出本效果的完整代码,如下:

package nd.no.xww.qqmessagedragview;import android.animation.ValueAnimator;
import android.annotation.SuppressLint;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.PointF;
import android.graphics.RectF;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.view.View;import java.util.Random;/*** @author xww* @desciption :* @date 2019/8/6* @time 12:11* 博主:威威喵* 博客:https://blog.csdn.net/smile_Running*/
public class ChargeBezierView extends View {private Paint mExternalPaint;private Paint mInnerPaint;private Paint mArcPaint;private Paint mCirclePaint;private Paint mTextPaint;private Paint mMaskPaint;private int mWidth;private int mHeight;// 充电进度值百分制private int mProgress;private float mArcProgress;private float mPaintSize;//水波纹于进度条的高度比private float rippleScale;//用于画进度private RectF mRect;private Random mRandom;private float mCircleX;private float mCircleY;private float mDefCircleRadius;// 对角线的长度private float mDiagonal;private boolean isFinished = false;//水波纹高度坐标private float x;private float y;private void init() {mExternalPaint = getPaint(Color.parseColor("#554F94CD"));mInnerPaint = getPaint(Color.parseColor("#66B8FF"));mArcPaint = getPaint(Color.parseColor("#7FFF00"));mArcPaint.setStyle(Paint.Style.STROKE);//空心mCirclePaint = getPaint(Color.parseColor("#F8F8FF"));mCirclePaint.setStyle(Paint.Style.STROKE);//空心mTextPaint = getPaint(Color.parseColor("#FF00ff"));mMaskPaint = getPaint(Color.parseColor("#FFFFFF"));mMaskPaint.setStyle(Paint.Style.STROKE);mRandom = new Random();mPaintSize = mTextPaint.getTextSize();}private Paint getPaint(int color) {Paint paint = new Paint();paint.setDither(true);paint.setAntiAlias(true);paint.setStrokeWidth(18f);paint.setTextSize(60f);paint.setColor(color);return paint;}public ChargeBezierView(Context context) {this(context, null);}public ChargeBezierView(Context context, @Nullable AttributeSet attrs) {this(context, attrs, 0);}public ChargeBezierView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);init();}@SuppressLint("DrawAllocation")@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);mWidth = MeasureSpec.getSize(widthMeasureSpec);mHeight = MeasureSpec.getSize(heightMeasureSpec);mCircleX = mWidth / 2;mCircleY = mHeight / 2;mDefCircleRadius = mWidth / 4;mRect = new RectF(mCircleX - mDefCircleRadius, mCircleY - mDefCircleRadius,mCircleX + mDefCircleRadius, mCircleY + mDefCircleRadius);mDiagonal = (float) Math.sqrt(Math.pow(mCircleX, 2) + Math.pow(mCircleY, 2));rippleScale = 2 * mDefCircleRadius / 100;}@Overrideprotected void onDraw(Canvas canvas) {if (isFinished) {drawMasked(canvas);drawFinished(canvas);} else {drawExternalRipple(canvas);drawMasked(canvas);drawProgressText(canvas);drawCircle(canvas);drawProgress(canvas);}}// 绘制电量圆形轨道private void drawCircle(Canvas canvas) {canvas.drawCircle(mCircleX, mCircleY, mDefCircleRadius, mCirclePaint);}private void drawProgress(Canvas canvas) {// -90 表示从上半轴 x=0 开始canvas.drawArc(mRect, -90, mArcProgress, false, mArcPaint);}private void drawProgressText(Canvas canvas) {canvas.drawText(mProgress + "%", mCircleX - mPaintSize, mCircleY + mTextPaint.getTextSize() / 2, mTextPaint);}private void drawMasked(Canvas canvas) {//绘制一个遮罩层,屏蔽 Path Close 以外的区域mMaskPaint.setStrokeWidth(mDiagonal + mDefCircleRadius * 2 - mPaintSize * 1.5f);canvas.drawCircle(mCircleX, mCircleY, mDiagonal, mMaskPaint);}private void drawFinished(Canvas canvas) {canvas.drawCircle(mCircleX, mCircleY, mDefCircleRadius, mArcPaint);canvas.drawCircle(mCircleX, mCircleY, mDefCircleRadius, mInnerPaint);canvas.drawText("充电完成", mCircleX - mTextPaint.getTextSize() * 2f, mCircleY + mTextPaint.getTextSize() / 2, mTextPaint);}private PointF pExt0;private PointF pExt1;private PointF pExt2;private PointF pExt3;private PointF pIn0;private PointF pIn1;private PointF pIn2;private PointF pIn3;ValueAnimator externalAnimator;// 绘制海浪的波纹效果,分内部和外部两条private void drawExternalRipple(Canvas canvas) {// 计算进度的 x , y 位置y = mCircleY - mDefCircleRadius + (100 - mProgress) * rippleScale;x = caculateX(y);float rippleY = y;float rippleX = mCircleX;//内部pIn0 = new PointF(rippleX - mDefCircleRadius, rippleY);pIn1 = new PointF(rippleX - mRandom.nextInt((int) mDefCircleRadius), rippleY - mRandom.nextInt((int) (mDefCircleRadius / 4)));pIn2 = new PointF(rippleX + mRandom.nextInt((int) mDefCircleRadius), rippleY + mRandom.nextInt((int) (mDefCircleRadius / 4)));pIn3 = new PointF(rippleX + mDefCircleRadius, rippleY);Path inPath = new Path();inPath.moveTo(pIn0.x, pIn0.y);inPath.cubicTo(pIn1.x, pIn1.y, pIn2.x, pIn2.y, pIn3.x, pIn3.y);inPath.lineTo(mCircleX + mDefCircleRadius, mCircleY + mDefCircleRadius);inPath.lineTo(mCircleX - mDefCircleRadius, mCircleY + mDefCircleRadius);inPath.close();canvas.drawPath(inPath, mInnerPaint);// 外部pExt0 = new PointF(rippleX - mDefCircleRadius, rippleY);pExt1 = new PointF(rippleX - mRandom.nextInt((int) mDefCircleRadius), rippleY + mRandom.nextInt((int) (mDefCircleRadius / 3)));pExt2 = new PointF(rippleX + mRandom.nextInt((int) mDefCircleRadius), rippleY + mRandom.nextInt((int) (mDefCircleRadius / 3)));pExt3 = new PointF(rippleX + mDefCircleRadius, rippleY);Path extPath = new Path();extPath.moveTo(pExt0.x, pExt0.y);extPath.cubicTo(pExt1.x, pExt1.y, pExt2.x, pExt2.y, pExt3.x, pExt3.y);extPath.lineTo(mCircleX + mDefCircleRadius, mCircleY + mDefCircleRadius);extPath.lineTo(mCircleX - mDefCircleRadius, mCircleY + mDefCircleRadius);extPath.close();canvas.drawPath(extPath, mExternalPaint);}public void setProgress(int progress) {this.mProgress = progress;this.mArcProgress = mProgress * 3.6f;if (mProgress <= 100) {isFinished = false;} else {isFinished = true;}invalidate();}// 圆的方程式 a2 = b2 + c2private float caculateX(float y) {x = (float) Math.sqrt(Math.pow(mDefCircleRadius, 2) - y * y);return x;}
}

    还有一个是进行进度值设置的,这个很简单,在 MainActivity 里面开一个子线程,然后设置一下进度值就可以了

        chargeView = findViewById(R.id.chargeView);new Thread(new Runnable() {@Overridepublic void run() {while (true) {progress++;if (progress > 100) {progress = 101;}try {Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}runOnUiThread(new Runnable() {@Overridepublic void run() {chargeView.setProgress(progress);}});}}}).start();

    使用起来就是这么简单,不过还有一些与贝塞尔曲线相关的知识没有介绍,感兴趣的话,可以去看我之前写的几篇文章,里面有关于贝塞尔的介绍,还有一些比较炫酷的 Android 动画效果哦。

这篇关于贝塞尔曲线(Bezier)之水波纹的手机充电动画效果(一)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/601123

相关文章

最好用的WPF加载动画功能

《最好用的WPF加载动画功能》当开发应用程序时,提供良好的用户体验(UX)是至关重要的,加载动画作为一种有效的沟通工具,它不仅能告知用户系统正在工作,还能够通过视觉上的吸引力来增强整体用户体验,本文给... 目录前言需求分析高级用法综合案例总结最后前言当开发应用程序时,提供良好的用户体验(UX)是至关重要

基于Python实现PDF动画翻页效果的阅读器

《基于Python实现PDF动画翻页效果的阅读器》在这篇博客中,我们将深入分析一个基于wxPython实现的PDF阅读器程序,该程序支持加载PDF文件并显示页面内容,同时支持页面切换动画效果,文中有详... 目录全部代码代码结构初始化 UI 界面加载 PDF 文件显示 PDF 页面页面切换动画运行效果总结主

React实现原生APP切换效果

《React实现原生APP切换效果》最近需要使用Hybrid的方式开发一个APP,交互和原生APP相似并且需要IM通信,本文给大家介绍了使用React实现原生APP切换效果,文中通过代码示例讲解的非常... 目录背景需求概览技术栈实现步骤根据 react-router-dom 文档配置好路由添加过渡动画使用

你的华为手机升级了吗? 鸿蒙NEXT多连推5.0.123版本变化颇多

《你的华为手机升级了吗?鸿蒙NEXT多连推5.0.123版本变化颇多》现在的手机系统更新可不仅仅是修修补补那么简单了,华为手机的鸿蒙系统最近可是动作频频,给用户们带来了不少惊喜... 为了让用户的使用体验变得很好,华为手机不仅发布了一系列给力的新机,还在操作系统方面进行了疯狂的发力。尤其是近期,不仅鸿蒙O

使用Python实现生命之轮Wheel of life效果

《使用Python实现生命之轮Wheeloflife效果》生命之轮Wheeloflife这一概念最初由SuccessMotivation®Institute,Inc.的创始人PaulJ.Meyer... 最近看一个生命之轮的视频,让我们珍惜时间,因为一生是有限的。使用python创建生命倒计时图表,珍惜时间

Qt QWidget实现图片旋转动画

《QtQWidget实现图片旋转动画》这篇文章主要为大家详细介绍了如何使用了Qt和QWidget实现图片旋转动画效果,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 一、效果展示二、源码分享本例程通过QGraphicsView实现svg格式图片旋转。.hpjavascript

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

防近视护眼台灯什么牌子好?五款防近视效果好的护眼台灯推荐

在家里,灯具是属于离不开的家具,每个大大小小的地方都需要的照亮,所以一盏好灯是必不可少的,每个发挥着作用。而护眼台灯就起了一个保护眼睛,预防近视的作用。可以保护我们在学习,阅读的时候提供一个合适的光线环境,保护我们的眼睛。防近视护眼台灯什么牌子好?那我们怎么选择一个优秀的护眼台灯也是很重要,才能起到最大的护眼效果。下面五款防近视效果好的护眼台灯推荐: 一:六个推荐防近视效果好的护眼台灯的

Flutter 进阶:绘制加载动画

绘制加载动画:由小圆组成的大圆 1. 定义 LoadingScreen 类2. 实现 _LoadingScreenState 类3. 定义 LoadingPainter 类4. 总结 实现加载动画 我们需要定义两个类:LoadingScreen 和 LoadingPainter。LoadingScreen 负责控制动画的状态,而 LoadingPainter 则负责绘制动画。

用Unity2D制作一个人物,实现移动、跳起、人物静止和动起来时的动画:中(人物移动、跳起、静止动作)

上回我们学到创建一个地形和一个人物,今天我们实现一下人物实现移动和跳起,依次点击,我们准备创建一个C#文件 创建好我们点击进去,就会跳转到我们的Vision Studio,然后输入这些代码 using UnityEngine;public class Move : MonoBehaviour // 定义一个名为Move的类,继承自MonoBehaviour{private Rigidbo