Android-动画的入门和实现

在我们Android开发中,经常会使用到简单或者动画,这里就对Android开发中的动画做了一下简单的总结。

Android 动画分类

总的来说,Android动画可以分为两类,最初的传统动画和Android3.0 之后出现的属性动画

传统动画又包括帧动画(Frame Animation)补间动画(Tweened Animation)

Paste_Image.png

一传统动画

1.帧动画

帧动画是最容易实现的一种动画,这种动画更多的依赖于完善的UI资源,他的原理就是将一张张单独的图片连贯的进行播放,

从而在视觉上产生一种动画的效果;而我们要做的就是:使用代码把一幅幅的图片按顺序显示。

我们来写上面逗比猫的动画:

<?xml version="1.0"encoding="utf-8"?>

protectedvoidonCreate(@Nullable Bundle savedInstanceState){super.onCreate(savedInstanceState);        setContentView(R.layout.activity_frame_animation);        ImageView animationImg1 = (ImageView) findViewById(R.id.animation1);        animationImg1.setImageResource(R.drawable.frame_anim1);        AnimationDrawable animationDrawable1 = (AnimationDrawable) animationImg1.getDrawable();        animationDrawable1.start();    }

在有些代码中,我们还会看到android:oneshot="false",这个oneshot 的含义就是动画执行一次(true)还是循环执行多次。

上面其他几个动画实现方式都是一样,无非就是图片资源的差异。(其实我们小时看过的动画片大部分就是这样搞出来的)

2.补间动画

补间动画又可以分为四种形式,分别是** alpha(淡入淡出)translate(位移)scale(缩放大小)rotate(旋转)**。

补间动画的实现,一般会采用xml 文件的形式;代码会更容易书写和阅读,同时也更容易复用。

XML 实现

首先,在res/anim/文件夹下定义如下的动画实现方式

alpha_anim.xml 动画实现

<?xml version="1.0"encoding="utf-8"?>

scale.xml 动画实现

<?xml version="1.0"encoding="utf-8"?>

然后,在Activity中

Animation animation = AnimationUtils.loadAnimation(mContext, R.anim.alpha_anim);

img = (ImageView) findViewById(R.id.img);

img.startAnimation(animation);

这样就可以实现ImageView alpha透明变化的动画效果。

也可以使用set 标签将多个动画组合(代码源自Android SDK API)

<?xml version="1.0"encoding="utf-8"?>...

可以看到组合动画是可以嵌套使用的。

各个动画属性的含义结合动画自身的特点应该很好理解,就不一一阐述了;这里主要说一下interpolatorpivot

Interpolator 主要作用是可以控制动画的变化速率 ,就是动画进行的快慢节奏。

Android 系统已经为我们提供了一些Interpolator ,比如 accelerate_decelerate_interpolator,accelerate_interpolator等。更多的interpolator 及其含义可以在Android SDK 中查看。同时这个Interpolator也是可以自定义的,这个后面还会提到。

pivot 决定了当前动画执行的参考位置

pivot 这个属性主要是在translate 和 scale 动画中,这两种动画都牵扯到view 的“物理位置“发生变化,所以需要一个参考点。而pivotX和pivotY就共同决定了这个点;它的值可以是float或者是百分比数值。

我们以pivotX为例,

pivotX取值含义

10距离动画所在view自身左边缘10像素

10%距离动画所在view自身左边缘 的距离是整个view宽度的10%

10%p距离动画所在view父控件左边缘的距离是整个view宽度的10%

pivotY 也是相同的原理,只不过变成的纵向的位置。如果还是不明白可以参考源码,在Tweened Animation中结合seekbar的滑动观察rotate的变化理解。

使用代码实现

有时候,动画的属性值可能需要动态的调整,这个时候使用xml 就不合适了,需要使用代码实现

privatevoidRotateAnimation(){        animation =newRotateAnimation(-deValue, deValue, Animation.RELATIVE_TO_SELF,                pxValue, Animation.RELATIVE_TO_SELF, pyValue);        animation.setDuration(timeValue);if(keep.isChecked()) {            animation.setFillAfter(true);        }else{            animation.setFillAfter(false);        }if(loop.isChecked()) {            animation.setRepeatCount(-1);        }else{            animation.setRepeatCount(0);        }if(reverse.isChecked()) {            animation.setRepeatMode(Animation.REVERSE);        }else{            animation.setRepeatMode(Animation.RESTART);        }        img.startAnimation(animation);    }

这里animation.setFillAfter决定了动画在播放结束时是否保持最终的状态;animation.setRepeatCount和animation.setRepeatMode 决定了动画的重复次数及重复方式,具体细节可查看源码理解。

好了,传统动画的内容就说到这里了,个人觉得传统动画真的是被时代的潮流推得很靠后了。

二.属性动画

属性动画是现在使用的最多的一种动画,顾名思义它是对于对象属性的动画。它比补间动画更加强大,所有补间动画的内容,都可以通过属性动画实现。

属性动画大致分为两种类型ViewPropertyAnimatorObjectAnimator

ViewPropertyAnimator:适合一些通用的动画,比如旋转、位移、缩放和透明,使用方式也很简单通过View.animate()即可得到ViewPropertyAnimator,之后进行相应的动画操作即可。

ObjectAnimator:适合用于为我们的自定义控件添加动画,当然首先我们应该在自定义View中添加相应的getXXX()和setXXX()相应属性的getter和setter方法,这里需要注意的是在setter方法内改变了自定义View中的属性后要调用invalidate()来刷新View的绘制。之后调用ObjectAnimator.of属性类型()返回一个ObjectAnimator,调用start()方法启动动画即可。

属性动画入门

首先我们来看看如何用属性动画实现上面补间动画的效果

privatevoidRotateAnimation(){    ObjectAnimator anim = ObjectAnimator.ofFloat(myView,"rotation",0f,360f);    anim.setDuration(1000);    anim.start();}privatevoidAlpahAnimation(){    ObjectAnimator anim = ObjectAnimator.ofFloat(myView,"alpha",1.0f,0.8f,0.6f,0.4f,0.2f,0.0f);    anim.setRepeatCount(-1);    anim.setRepeatMode(ObjectAnimator.REVERSE);    anim.setDuration(2000);    anim.start();}

这两个方法用属性动画的方式分别实现了旋转动画和淡入淡出动画,其中setDuration、setRepeatMode及setRepeatCount和补间动画中的概念是一样的。

可以看到,属性动画貌似强大了许多,实现很方便,同时动画可变化的值也有了更多的选择,动画所能呈现的细节也更多。

当然一样牛逼的是属性动画也是可以组合实现

ObjectAnimator alphaAnim = ObjectAnimator.ofFloat(myView,"alpha",1.0f,0.5f,0.8f,1.0f);                ObjectAnimator scaleXAnim = ObjectAnimator.ofFloat(myView,"scaleX",0.0f,1.0f);                ObjectAnimator scaleYAnim = ObjectAnimator.ofFloat(myView,"scaleY",0.0f,2.0f);                ObjectAnimator rotateAnim = ObjectAnimator.ofFloat(myView,"rotation",0,360);                ObjectAnimator transXAnim = ObjectAnimator.ofFloat(myView,"translationX",100,400);                ObjectAnimator transYAnim = ObjectAnimator.ofFloat(myView,"tranlsationY",100,750);                AnimatorSetset=newAnimatorSet();set.playTogether(alphaAnim, scaleXAnim, scaleYAnim, rotateAnim, transXAnim, transYAnim);//                set.playSequentially(alphaAnim, scaleXAnim, scaleYAnim, rotateAnim, transXAnim, transYAnim);set.setDuration(3000);set.start();

可以看到这些动画可以同时播放,或者是按序播放。

动画监听器

您可以在动画持续时间期间使用下面描述的侦听器侦听重要事件。

Animator.AnimatorListener

onAnimationStart() - 动画启动时调用。

onAnimationEnd() - 当动画结束时调用。

onAnimationRepeat() - 当动画重演调用。

onAnimationCancel()-当动画被取消调用。取消的动画还呼吁onAnimationEnd(),无论他们是如何结束。

ValueAnimator.AnimatorUpdateListener

onAnimationUpdate()-称为动画的每一帧上。 听此事件使用所产生的计算值ValueAnimator动画中。 要使用该值,查询ValueAnimator传递到事件来获取与当前动画值对象getAnimatedValue()方法。 实现这个监听器。

根据什么属性或对象的动画,你可能需要调用invalidate()上查看迫使屏幕的该区域的新的动画值重绘自身。 例如,对Drawable对象的颜色属性进行动画处理只会在该对象重绘本身时更新屏幕。 所有在查看属性制定者,如setAlpha()和setTranslationX()正确无效视图,所以你不需要调用这些方法与新的值时无效视图。

在XML声明动画

属性动画系统允许您使用XML声明属性动画,而不是以编程方式。 通过在XML中定义动画,您可以轻松地在多个活动中重复使用动画,并更轻松地编辑动画序列。

为了区分使用与那些使用旧的新的属性动画的API动画文件观看动画框架,开始与Android 3.1,你应该保存XML文件在属性动画res/animator/目录下。

以下属性动画类通过以下XML标记支持XML声明:

ValueAnimator -

ObjectAnimator -

AnimatorSet -

下面的示例播放两套对象的动画顺序,第一组嵌套组一起编写两个对象的动画:

为了运行这个动画,必须在代码中的充气XML资源AnimatorSet对象,然后开始动画集之前为所有动画的目标对象。呼叫setTarget()设置的所有儿童的单一目标对象AnimatorSet的方便。下面的代码演示如何做到这一点:

AnimatorSetset= (AnimatorSet) AnimatorInflater.loadAnimator(myContext,  R.anim.property_animator);set.setTarget(myObject);set.start();

当然了,关于动画已经有好多NB的开源动画库了例如Lottie动画库Facebook-Rebound 弹性动画库

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,335评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,895评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,766评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,918评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,042评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,169评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,219评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,976评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,393评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,711评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,876评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,562评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,193评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,903评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,699评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,764评论 2 351

推荐阅读更多精彩内容

  • 1 背景 不能只分析源码呀,分析的同时也要整理归纳基础知识,刚好有人微博私信让全面说说Android的动画,所以今...
    未聞椛洺阅读 2,699评论 0 10
  • 在我们Android开发中,经常会使用到简单或者动画,这里就对Android开发中的动画做了一下简单的总结。 An...
    夏尼采阅读 360评论 0 11
  • 【Android 动画】 动画分类补间动画(Tween动画)帧动画(Frame 动画)属性动画(Property ...
    Rtia阅读 6,123评论 1 38
  • 一: 传统 View 动画(Tween/Frame) 1.1 Tween 动画 主要有 4 中:缩放、平移、渐变、...
    dfg_fly阅读 714评论 1 2
  • Animation Animation类是所有动画(scale、alpha、translate、rotate)的基...
    四月一号阅读 1,906评论 0 10