热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

仿华为02旋转圆球(圆点)和对勾

最终效果和原理小球位置和动画的实现原理小球半径的计算对勾的实现 代码 最终效果和原理 效果分为三方面多个小球按上图动画旋转 小球的大小按规律变化,模拟分散和聚合 最后画一个对勾 小球位置和

      • 最终效果和原理
        • 小球位置和动画的实现原理
        • 小球半径的计算
        • 对勾的实现
      • 代码


最终效果和原理

gif图片出错

效果分为三方面
多个小球按上图动画旋转
小球的大小按规律变化,模拟分散和聚合
最后画一个对勾


小球位置和动画的实现原理

本篇文章用到了仿华为01-圆环进度和小球 里面的知识点:属性动画和插值器
上面的文章中只有一个小球.我们把他看成是此处最前面的那个小球.小球做加速减速动画.

小球动画实现的难点在于各个小球 位置的确定和变化
参考第 1 个小球实现的原理.可以想到,后面的小球也需要做加速减速动画.
但是加速时要依次递减. 也就是第 1 小球的加速度>第 2 个小球>第 3 个小球>…这样小球就会渐渐分散开
减速时也要依次递减 也就是第 1 个小球减速最快,后面的减速慢.这样后面的小球会追上前面的,形成类似融合效果.

总结: 每个小球都做加速减速动画.用不同的加速度和减速度来实现分散和聚合效果.不同的加速度和减速度我们可以用幂函数来实现.

实现方式:
为每个小球设置一个属性动画,并自定义对应的插值器.
只给第 1 个小球设置属性动画,其余小球位置由第一个小球确定

为了减少代码和方便理解.我们用第二种方法实现

第1个第2个第3个第4个第5个
比例001.201.201.501.501.901.902.502.5
实际角度00000
比例0.1250.1251.20.1251.20.1251.50.1251.50.1251.90.1251.90.1252.50.1252.5
实际角度45301672
比例0.250.251.20.251.20.251.50.251.50.251.90.251.90.252.50.252.5
实际角度9068304526
比例0.50.51.20.51.20.51.50.51.50.51.90.51.90.52.50.52.5
实际角度1801561279663
比例0.750.751.20.751.20.751.50.751.50.751.90.751.90.752.50.752.5
实际角度270254233208175
比例111.211.211.511.511.911.912.512.5
实际角度360360360360360

总结:
1. 以第 1 个小球(角度占360度比例)为幂函数底数,后面小球对应的幂为 1.2, 1.5, 1.9, 2.5
2. 因为第 1 个小球本身具有加速减速动画,以其为基准后所有小球就用相同动画特点
3. 起始,结束位置相同,速度变化不同,实现了小球间分散聚合

小球半径的计算


为使得聚散效果更逼真,小球的半径还需要变化.理想状态是0-180度时,前面小球变小,后面小球变大,仿佛是前面分解了部分融入后面. 180-360则相反.整个过程保持前面小球半径大于后面小球.
这里没能实现理想效果,小球半径是一起变大和变小的.

小球 1 半径的变化规律
这里写图片描述
上图是小球 1 的半径变化示意图,起始结束位置最大,180度时半径减半
假设小球 1 此时半径为 r , 小球 2 就为 r /1.2, 小球 3 是 r/1.5 …. 具体可以自己调整.

对勾的实现

画对勾是 View 对外暴露的方法.当需要的时候调用. 此时结束小球动画,设置画笔透明度setAlpha() 来画出一个缓慢浮现的对勾

代码

public class DotRotateAndCheckMark extends View {Paint mPaint;float mWidth;float mHeight;// 作图的最小范围float minSize;// 第一个圆旋转的角度float angle;// 第一个圆最大半径float maxRadius;float[] radio &#61; new float[5];ObjectAnimator circleAnimator;// 画对号相关参数boolean isDrawMark &#61; false;Path path;int mMarkAlpha;public float getAngle() {return angle;}public void setAngle(float angle) {this.angle &#61; angle;}public DotRotateAndCheckMark(Context context) {this(context, null);}public DotRotateAndCheckMark(Context context, &#64;Nullable AttributeSet attrs) {this(context, attrs, 0);}public DotRotateAndCheckMark(Context context, &#64;Nullable AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);init();}private void init() {mPaint &#61; new Paint(Paint.ANTI_ALIAS_FLAG);mPaint.setStrokeWidth(4);mPaint.setStyle(Paint.Style.FILL);mPaint.setColor(Color.BLUE);postDelayed(new Runnable() {&#64;Overridepublic void run() {startCircleRun();}}, 800);radio[0] &#61; 1f;radio[1] &#61; 1.2f;radio[2] &#61; 1.5f;radio[3] &#61; 1.9f;radio[4] &#61; 2.5f;path &#61; new Path();}&#64;Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);mWidth &#61; getWidth();mHeight &#61; getHeight();minSize &#61; Math.min(mWidth, mHeight) * 0.25f;maxRadius &#61; minSize * 0.06f;canvas.translate(mWidth / 2, mHeight / 2);if (isDrawMark) {drawMark(canvas);} else {drawDots(canvas);}}private void drawMark(Canvas canvas) {mPaint.setStyle(Paint.Style.STROKE);mPaint.setStrokeCap(Paint.Cap.ROUND);mPaint.setStrokeJoin(Paint.Join.ROUND);path.moveTo(-minSize * 0.5f, -minSize * 0.15f);path.lineTo(-minSize * 0.05f, minSize * 0.3f);path.lineTo(minSize * 0.6f, -minSize * 0.4f);mPaint.setAlpha(mMarkAlpha);canvas.drawPath(path, mPaint);}private void drawDots(Canvas canvas) {mPaint.setStyle(Paint.Style.FILL);for (int i &#61; 0; i <5; i&#43;&#43;) {canvas.save();canvas.rotate((float) (Math.pow(angle / 360, radio[i]) * 360), 0, 0);canvas.drawCircle(0, -minSize, getRadius(angle, i), mPaint);canvas.restore();}//postDelayed(new Runnable() {&#64;Overridepublic void run() {invalidate();}}, 20);}// 获取小球半径private float getRadius(float angele, int pos) {float radius;if (angele <&#61; 180) {radius &#61; (1 - (angele / 360)) * maxRadius;} else {radius &#61; (angele / 360) * maxRadius;}return radius / radio[pos];}// 开始小球动画private void startCircleRun() {// 控制 angle 参数变化的属性动画if (circleAnimator &#61;&#61; null) {circleAnimator &#61; ObjectAnimator.ofFloat(this, "angle", 360);circleAnimator.setDuration(1500);circleAnimator.setRepeatCount(ValueAnimator.INFINITE);circleAnimator.setInterpolator(new AccelerateDecelerateInterpolator());circleAnimator.start();// 动画开始后,setAngele()会被反射调用从而设置 angele 值} else {circleAnimator.cancel();circleAnimator.start();}}// 开始对勾public void drawCheckMark() {if (circleAnimator !&#61; null) circleAnimator.cancel();isDrawMark &#61; true;ValueAnimator animator &#61; ValueAnimator.ofInt(0, 255);animator.setDuration(2500);animator.setInterpolator(new AccelerateDecelerateInterpolator());animator.start();animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {&#64;Overridepublic void onAnimationUpdate(ValueAnimator animation) {mMarkAlpha &#61; (int) animation.getAnimatedValue();invalidate();}});}
}



备注
AndroidStudio 3.1.2, compileSdkVersion 27
小球的属性动画作用于属性 angle,对应 set() get() 方法不可或缺
小球动画是自动开启,画对勾方法对外暴露
没有对外提供 attr 属性设置
项目地址




不足
小球半径变化未能实现理想状态
小球变对号之间没有转场动画


推荐阅读
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • Linux重启网络命令实例及关机和重启示例教程
    本文介绍了Linux系统中重启网络命令的实例,以及使用不同方式关机和重启系统的示例教程。包括使用图形界面和控制台访问系统的方法,以及使用shutdown命令进行系统关机和重启的句法和用法。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 个人学习使用:谨慎参考1Client类importcom.thoughtworks.gauge.Step;importcom.thoughtworks.gauge.T ... [详细]
  • 如何自行分析定位SAP BSP错误
    The“BSPtag”Imentionedintheblogtitlemeansforexamplethetagchtmlb:configCelleratorbelowwhichi ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
author-avatar
天雨
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有