Android自定义view实现阻尼效果的加载动画

Android自定义view实现阻尼效果的加载动画,第1张

概述效果:需要知识:1.二次贝塞尔曲线2.动画知识3.基础自定义view知识先来解释下什么叫阻尼运动

效果:

需要知识:

1. 二次贝塞尔曲线

2. 动画知识

3. 基础自定义view知识

先来解释下什么叫阻尼运动

阻尼振动是指,由于振动系统受到摩擦和介质阻力或其他能耗而使振幅随时间逐渐衰减的振动,又称减幅振动、衰减振动。[1] 不论是d簧振子还是单摆由于外界的摩擦和介质阻力总是存在,在振动过程中要不断克服外界阻力做功,消耗能量,振幅就会逐渐减小,经过一段时间,振动就会完全停下来。这种振幅随时间减小的振动称为阻尼振动.因为振幅与振动的能量有关,阻尼振动也就是能量不断减少的振动.阻尼振动是非简谐运动.阻尼振动系统属于耗散系统。这里的阻尼是指任何振动系统在振动中,由于外界作用或系统本身固有的原因引起的振动幅度逐渐下降的特性,以及此一特性的量化表征。

本例中文字部分凹陷就是这种效果,当然这篇文章知识带你简单的使用.

跳动的水果效果实现

剖析:从上面的效果图中很面就是从顶端向下掉落然后再向上 期间旋转即可.

那么我们首先自定义一个vIEw继承FrameLayout

public class My extends FrameLayout {public My(Context context) {super(context);}public My(Context context,AttributeSet attrs) {super(context,attrs);}}

需要素材如下三张图片:



也许有人会问我看到你效果图到顶部或者底部就变成向上或者向下了.你三张够吗?

答:到顶部或者底部旋转180度即可

我们现在自定义中定义几个变量

//用于记录当前图片使用数组中的哪张int indeximgFlag = 0;//下沉图片 前面三个图片的IDint allimgDown [] = {R.mipmap.p2,R.mipmap.p4,R.mipmap.p6,R.mipmap.p8};//动画效果一次下沉或上d的时间 animationDuration*2=一次完整动画时间int animationDuration = 1000;//d起来的图片ImageVIEw iv;//图片下沉高度(即从最高点到最低点的距离)int downHeight = 2;//掉下去的动画private Animation translateDown;//d起动画private Animation translateUp;//旋转动画private ObjectAnimator rotation;

我们再来看看初始化动画的方法(此方法使用了递归思想,实现无限播放动画,大家可以看看哪里不理解)

//初始化d跳动画public voID MyAnmation(){//下沉效果动画translateDown = new TranslateAnimation(Animation.relative_TO_SELF,Animation.relative_TO_SELF,downHeight);translateDown.setDuration(animationDuration);//设置一个插值器 动画将会播放越来越快 模拟重力translateDown.setInterpolator(new AccelerateInterpolator());//上d动画translateUp = new TranslateAnimation(Animation.relative_TO_SELF,downHeight,0);translateUp.setDuration(animationDuration);////设置一个插值器 动画将会播放越来越慢 模拟反重力translateUp.setInterpolator(new DecelerateInterpolator());//当下沉动画完成时播放启动上dtranslateDown.setAnimationListener(new Animation.AnimationListener() {@OverrIDepublic voID onAnimationStart(Animation animation) {iv.setimageResource(allimgDown[indeximgFlag]);rotation = ObjectAnimator.offloat(iv,"rotation",180f,360f);rotation.setDuration(1000);rotation.start();}@OverrIDepublic voID onAnimationEnd(Animation animation) {iv.startAnimation(translateUp);}@OverrIDepublic voID onAnimationRepeat(Animation animation) {}});//当上移动画完成时 播放下移动画translateUp.setAnimationListener(new Animation.AnimationListener() {@OverrIDepublic voID onAnimationStart(Animation animation) {indeximgFlag = 1+indeximgFlag>=allimgDown.length?0:1+indeximgFlag;iv.setimageResource(allimgDown[indeximgFlag]);rotation = ObjectAnimator.offloat(iv,0.0f,180f);rotation.setDuration(1000);rotation.start();}@OverrIDepublic voID onAnimationEnd(Animation animation) {//递归iv.startAnimation(translateDown);}@OverrIDepublic voID onAnimationRepeat(Animation animation) {}});}

以上代码知识:

插值器:会让一个动画在播放时在某一时间段加快动画或者减慢

//设置一个插值器 动画将会播放越来越快 模拟重力

1.translateDown.setInterpolator(new AccelerateInterpolator());

这个插值器 速率表示图:

可以从斜率看到使用此插值器 动画将越来越快.意义在于模仿下落时重力的影响

////设置一个插值器 动画将会播放越来越慢 模拟反重力

2. translateUp.setInterpolator(new DecelerateInterpolator());

速率图:

最后我们初始化下图片控件到我们的自定义view

private voID init() {//初始化d跳图片 控件iv = new ImageVIEw(getContext());VIEwGroup.LayoutParams params = new VIEwGroup.LayoutParams(VIEwGroup.LayoutParams.WRAP_CONTENT,VIEwGroup.LayoutParams.WRAP_CONTENT);iv.setLayoutParams(params);iv.setimageResource(allimgDown[0]);this.addVIEw(iv);iv.measure(0,0);iv.getVIEwTreeObserver().addOnGlobalLayoutListener(new VIEwTreeObserver.OnGlobalLayoutListener() {@OverrIDepublic voID onGlobalLayout() {if (!flagMeure){flagMeure =true;//由于画文字是由基准线开始path.moveto(iv.getX()-textWIDth/2+iv.getWIDth()/2,textHeight+iv.getHeight()+downHeight*iv.getHeight());//计算最大d力maxElasticFactor = (float) (textHeight / elastic);//初始化贝塞尔曲线path.rQuadTo(textWIDth / 2,textWIDth,0);//初始化上d和下沉动画MyAnmation();iv.startAnimation(translateDown);}}});}

上面的知识:

1. iv.measure(0,0);主动通知系统去测量此控件 不然iv.getwIDth = 0;//下面这个是同理 等iv测量完时回调 不然iv.getwIDth = 0;2. iv.getVIEwTreeObserver().addOnGlobalLayoutListener(new VIEwTreeObserver.OnGlobalLayoutListener(){ … }

原因:TextVIEw tv = new TextVIEw() 或者 LayoutInflat 填充布局都是

异步所以你在new出来或者填充时直接获取自然返回0

到现在为止你只需要在自定义view 的onSizeChanged回调方法中调用init()即可看到动画的d动

@OverrIDeprotected voID onSizeChanged(int w,int h,int olDW,int oldh) {super.onSizeChanged(w,h,olDW,oldh);init();}

此方法会在onmesure方法执行完成后回调 这样你就可以在此方法获得自定义view的宽高了

效果图:

画文字成u形

首先你得知道如下知识

贝塞尔曲线:具体学习

这里我们用到2此贝塞尔曲线

我们看看大概是什么叫2次贝塞尔曲线

 

我们看看 三个点 p0 p1 p2 我们 把p0 称为 开始点 p1 为控制点 p2 结束点,那么可以用贝塞尔公式画出如图曲线

这里大家没必要深究怎么画出来. 也不需要你懂 这个要数学基础的

那么我们在安卓中怎么画呢?

Path path = new Path();//p0的x y坐标path.moveto(p0.x,y);path.rQuadTo(p1.x,p1.y,p2.x,p2.y);

这就是API调用方法是不是很简单?那么你又会问那么怎么画出来呢?
很简单在 dispatchDraw方法 或者onDraw中 调用

@OverrIDeprotected voID dispatchDraw(Canvas canvas) {super.dispatchDraw(canvas);canvas.drawPath(path,paint);}

那么你画出来的效果应该和在Ps用钢笔画出来的差不多 ps中钢笔工具就是二次贝塞尔曲线

 

(借用下图片)

如果你的三个点的位置如刚开的图片 p0 p1 p2 (p1在p0右上方并且 p1在p2左上方)一样那么在屏幕中的显示效果如下

 

这里随扩张下dispatchDraw和ondraw的区别

如果你的自定义view 是继承vIEw 那么 会先调用 ondraw->>dispatchDraw

如果你的自定义view 是继承vIEwgroup那么会跳过ondraw方法直接调用dispathchDraw这里特别注意!!我们这个案例中继承的是FrameLayout,而frameLayout又是继承自vIEwgroup所以….

那么我们回到主题 如何画一个U形文字?简单就是说按照我们画出来的曲线在上面写字 如: 文字是”CSDN开源中国” 如何让这几个字贴着我们的曲线写出来?

这里介绍一个API

canvas.drawTextOnPath()

 

第一个参数:文字 类型为字符串

第二个参数:路径 也就是我们前面的二次贝塞尔曲线

第三个参数:沿着路径文字开始的位置 说白了偏移量

第四个参数:贴着路径的高度的偏移量

hOffset:

The distance along the path to add to the text's starting position

vOffset:

The distance above(-) or below(+) the path to position the text
//ok我们看看他可以画出什么样子的文字

这种看大家对贝塞尔曲线的理解,你理解的越深那么你可以画出的图像越多,当然不一定要用贝塞尔曲线

确定贝塞尔曲线的起点

我们在回过头来看看我们的效果图

我们可以看到文字应该是在iv(d跳的图片中央位置且正好在 ivd到底部的位置)

这里我们先补充知识在考虑计算

我们来学习一下文字的测量我们来看幅图

我们调用画文字的API时

canvas.drawTextOnPath或者canvas.drawText 是从基准线开始画的也就是说途中的baseline开始画.

如:

canvas.drawText(“FMY”,paint);

那么你将看不到文字 只能在屏幕看到文字底部如下图:

另一个同理API drawTextOnPath 也是

再看看几个简单的API

1 . paint.measureText(“FMY”);返回在此画笔paint下写FMY文字的宽度

下面的API会把文字的距离左边left 上边top 右边right 底部的bottom的值写入此矩形 那么

rect.right-rect.left=文字宽度 rect.bottom-rect.top=文字高度//矩形Rect rect = new Rect();//将文字画入矩形目的是为了测量高度paint.getTextBounds(printText,printText.length(),rect);

那么请看:

private voID init() {//初始化d跳图片 控件iv = new ImageVIEw(getContext());VIEwGroup.LayoutParams params = new VIEwGroup.LayoutParams(VIEwGroup.LayoutParams.WRAP_CONTENT,VIEwGroup.LayoutParams.WRAP_CONTENT);iv.setLayoutParams(params);iv.setimageResource(allimgDown[0]);this.addVIEw(iv);//画笔的初始化paint = new Paint();paint.setstrokeWIDth(1);paint.setcolor(color.CYAN);paint.setStyle(Paint.Style.FILL);paint.setTextSize(50);paint.setAntiAlias(true);//矩形Rect rect = new Rect();//将文字画入矩形目的是为了测量高度paint.getTextBounds(printText,rect);//文本宽度textWIDth = paint.measureText(printText);//获得文字高度textHeight = rect.bottom - rect.top;//初始化路径path = new Path();iv.setX(getWIDth()/2);iv.measure(0,0);//初始化上d和下沉动画MyAnmation();iv.startAnimation(translateDown);}}});}

我们现在写一个类当iv图片(d跳图)碰到文字顶部时设置一个监听器 时间正好是d图向上到顶部的时间 期间不断让文字凹陷在恢复正常

//用于播放文字下沉和上浮动画传入的数值必须是 图片下沉和上升的一次时间public voID initAnimation(int duration){//这里为什maxElasticFactor/4 好看...另一个同理 这个数值大家自行调整ValueAnimator animator = ValueAnimator.offloat(maxElasticFactor/4,(float) (maxElasticFactor / 1.5),0);animator.setDuration(duration/2);animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {@OverrIDepublic voID onAnimationUpdate(ValueAnimator animation) {calc();//重新画路径NowElasticFactor= (float) animation.getAnimatedValue();postInvalIDate();}});animator.start();}

再来一个重新绘画路径计算的方法

public voID calc(){//重置路径path.reset();//由于画文字是由基准线开始path.moveto(iv.getX()-textWIDth/2+iv.getWIDth()/2,textHeight+iv.getHeight()+downHeight*iv.getHeight());//画二次贝塞尔曲线path.rQuadTo(textWIDth / 2,NowElasticFactor,0);}

好了到这里我们看看完整源码吧:

package com.example.administrator.myapplication;import androID.animation.ObjectAnimator;import androID.animation.ValueAnimator;import androID.content.Context;import androID.graphics.Canvas;import androID.graphics.color;import androID.graphics.Paint;import androID.graphics.Path;import androID.graphics.Rect;import androID.util.AttributeSet;import androID.vIEw.VIEwGroup;import androID.vIEw.VIEwTreeObserver;import androID.vIEw.animation.AccelerateInterpolator;import androID.vIEw.animation.Animation;import androID.vIEw.animation.DecelerateInterpolator;import androID.vIEw.animation.TranslateAnimation;import androID.Widget.FrameLayout;import androID.Widget.ImageVIEw;public class My extends FrameLayout {//画笔private Paint paint;//路径private Path path;//要输入的文本private String printText = "正在加载";//文本宽private float textWIDth;//文本高private float textHeight;//测量文字宽高的时候使用的矩形private Rect rect;//最大d力系数private float elastic = 1.5f;//最大d力private float maxElasticFactor;//当前d力private float NowElasticFactor;//用于记录当前图片使用数组中的哪张int indeximgFlag = 0;//下沉图片int allimgDown [] = {R.mipmap.p2,R.mipmap.p8};//动画效果一次下沉或上d的时间 animationDuration*2=一次完整动画时间int animationDuration = 1000;//d起来的图片ImageVIEw iv;//图片下沉高度(即从最高点到最低点的距离)int downHeight = 2;private Animation translateDown;private Animation translateUp;private ObjectAnimator rotation;public My(Context context) {super(context);}public My(Context context,attrs);}@OverrIDeprotected voID dispatchDraw(Canvas canvas) {super.dispatchDraw(canvas);canvas.drawTextOnPath(printText,path,paint);}//用于播放文字下沉和上浮动画传入的数值必须是 图片下沉和上升的一次时间public voID initAnimation(int duration){//这里为什maxElasticFactor/4为什么ValueAnimator animator = ValueAnimator.offloat(maxElasticFactor/4,0);animator.setDuration(duration/2);animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {@OverrIDepublic voID onAnimationUpdate(ValueAnimator animation) {calc();NowElasticFactor= (float) animation.getAnimatedValue();postInvalIDate();}});animator.start();}public voID calc(){//重置路径path.reset();//由于画文字是由基准线开始path.moveto(iv.getX()-textWIDth/2+iv.getWIDth()/2,0);}//初始化d跳动画public voID MyAnmation(){//下沉效果动画translateDown = new TranslateAnimation(Animation.relative_TO_SELF,360f);rotation.setDuration(1000);rotation.start();}@OverrIDepublic voID onAnimationEnd(Animation animation) {iv.startAnimation(translateUp);initAnimation(animationDuration);}@OverrIDepublic voID onAnimationRepeat(Animation animation) {}});//当上移动画完成时 播放下移动画translateUp.setAnimationListener(new Animation.AnimationListener() {@OverrIDepublic voID onAnimationStart(Animation animation) {indeximgFlag = 1+indeximgFlag>=allimgDown.length?0:1+indeximgFlag;iv.setimageResource(allimgDown[indeximgFlag]);rotation = ObjectAnimator.offloat(iv,180f);rotation.setDuration(1000);rotation.start();}@OverrIDepublic voID onAnimationEnd(Animation animation) {//递归iv.startAnimation(translateDown);}@OverrIDepublic voID onAnimationRepeat(Animation animation) {}});}boolean flagMeure;@OverrIDeprotected voID onSizeChanged(int w,oldh);init();}private voID init() {//初始化d跳图片 控件iv = new ImageVIEw(getContext());VIEwGroup.LayoutParams params = new VIEwGroup.LayoutParams(VIEwGroup.LayoutParams.WRAP_CONTENT,VIEwGroup.LayoutParams.WRAP_CONTENT);iv.setLayoutParams(params);iv.setimageResource(allimgDown[0]);this.addVIEw(iv);//画笔的初始化paint = new Paint();paint.setstrokeWIDth(1);paint.setcolor(color.CYAN);paint.setStyle(Paint.Style.FILL);paint.setTextSize(50);paint.setAntiAlias(true);//矩形rect = new Rect();//将文字画入矩形目的是为了测量高度paint.getTextBounds(printText,0);//初始化上d和下沉动画MyAnmation();iv.startAnimation(translateDown);}}});}}

小人奉上源码一封供大家 参考github源码下载地址

以上所述是小编给大家介绍的AndroID自定义view实现阻尼效果的加载动画,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对编程小技巧网站的支持!

总结

以上是内存溢出为你收集整理的Android自定义view实现阻尼效果的加载动画全部内容,希望文章能够帮你解决Android自定义view实现阻尼效果的加载动画所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1147777.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-31
下一篇 2022-05-31

发表评论

登录后才能评论

评论列表(0条)

保存