在Android里面,一些炫酷的动画确实是很吸引人的地方,让然看了就赏心悦目,一个好看的动画可能会提高用户对软件的使用率。另外说到动画,在Android里面支持两种动画:补间动画和属性动画,至于这两种动画的区别这里不再介绍,希望开发者都能在使用的过程中体会两者的不同。
本文使用属性动画完成,说到属性动画,肯定要提到 JakeWharton大神写的NineOldAndroids动画库,如果你的app需要在android3.0以下使用属性动画,那么这个库就很有作用了,如果只需要在高版本使用,那么直接使用系统提供的动画API即可。
首先看一下本文要实现的动画效果:手指向上移动到开关按钮处, 然后一个点击动作,开关从关到开动画执行,同时手指向下移动回到原来的位置
点击图片调转到对应链接查看动画
动画的使用场景
引导用户去打开某个功能的开关按钮或者去打开系统的某项设置的时候,增加动画可以提高用户的点击率,表达的意思也更明确
实现之前先做好如下准备工作
1. 下载nineoldandroids-2.4.0.jar的库,放到android studio 工程目录的libs文件夹中
2. 在build.gradle文件中引入
dependencies { compile files('libs/nineoldandroids-2.4.0.jar') }
3. 准备好相关的图片资源
接下来封装一个自定义控件来实现整个动画
第一步:先定义一个布局文件finger_switch_on_guide_layout.xml
<?xml version="1.0" encoding="utf-8"?> <merge xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/switch_anim_root" android:layout_width="wrap_content" android:layout_height="wrap_content"> <FrameLayout android:layout_width="wrap_content" android:layout_height="wrap_content"> <ImageView android:layout_width="42dp" android:layout_height="25dp" android:background="@drawable/switch_container" /> <ImageView android:id="@+id/switch_anim_circle_point" android:layout_width="20dp" android:layout_height="20dp" android:layout_marginLeft="2.5dp" android:layout_marginTop="2.5dp" android:background="@drawable/switch_off_circle_point" /> </FrameLayout> <ImageView android:id="@+id/finger_switch" android:layout_width="34dp" android:layout_height="41dp" android:layout_marginLeft="5dp" android:layout_marginTop="25dp" android:background="@drawable/finger_normal" /> </merge>
布局文件预缆长这样:
第二步:定义自定义控件(SwitchOnAnimView)实现整个动画
package com.androidanimation.animationview; import android.content.Context; import android.os.Handler; import android.util.AttributeSet; import android.view.LayoutInflater; import android.widget.FrameLayout; import android.widget.ImageView; import com.androidanimation.R; import com.androidanimation.animations.BaseAnimatorListener; import com.androidanimation.utils.ViewUtil; import com.nineoldandroids.animation.Animator; import com.nineoldandroids.animation.ObjectAnimator; import com.nineoldandroids.view.ViewHelper; /** * Created by popfisher on 2016/9/3. */ public class SwitchOnAnimView extends FrameLayout { private Handler mHandler = new Handler(); /** 开关中间的圆圈View */ private ImageView mCirclePtImgv; /** 手指View */ private ImageView mFingerImgv; /** 手指移动的距离 */ private float mFingerMoveDistance; /** 开关中间的圆圈View需要移动的距离 */ private float mCirclePtMoveDistance; private static final int FINGER_ANIM_DURATION = 300; private static final int CIRCLE_PT_ANIM_DURATION = 500; private boolean isStopAnim = false; public SwitchOnAnimView(Context context) { this(context, null); } public SwitchOnAnimView(Context context, AttributeSet attrs) { super(context, attrs); // 加载布局 LayoutInflater.from(context).inflate(R.layout.finger_switch_on_guide_layout, this, true); initView(); } private void initView() { mCirclePtImgv = (ImageView) findViewById(R.id.switch_anim_circle_point); mFingerImgv = (ImageView) findViewById(R.id.finger_switch); // 下面两个距离要根据UI布局来确定 mFingerMoveDistance = ViewUtil.dp2px(getContext(), 20f); mCirclePtMoveDistance = ViewUtil.dp2px(getContext(), 17.5f); } /** * 启动动画 */ public void startAnim() { isStopAnim = false; // 启动动画之前先恢复初始状态 ViewHelper.setTranslationX(mCirclePtImgv, 0); mCirclePtImgv.setBackgroundResource(R.drawable.switch_off_circle_point); mFingerImgv.setBackgroundResource(R.drawable.finger_normal); startFingerUpAnim(); } /** * 停止动画 */ public void stopAnim() { isStopAnim = true; } /** * 中间的圈点View平移动画 */ private void startCirclePointAnim() { if (mCirclePtImgv == null) { return; } ObjectAnimator circlePtAnim = ObjectAnimator.ofFloat(mCirclePtImgv, "translationX", 0, mCirclePtMoveDistance); circlePtAnim.setDuration(CIRCLE_PT_ANIM_DURATION); circlePtAnim.start(); } /** * 手指向上移动动画 */ private void startFingerUpAnim() { ObjectAnimator fingerUpAnim = ObjectAnimator.ofFloat(mFingerImgv, "translationY", 0, -mFingerMoveDistance); fingerUpAnim.setDuration(FINGER_ANIM_DURATION); fingerUpAnim.addListener(new BaseAnimatorListener() { @Override public void onAnimationEnd(Animator animator) { if (mFingerImgv == null || mHandler == null) { return; } // 手指向上动画执行完成就设置手指View背景为点击状态的背景 mFingerImgv.setBackgroundResource(R.drawable.finger_click); // 点击之后为了提现停顿一下的感觉,延迟200毫秒执行其他动画 mHandler.postDelayed(new Runnable() { @Override public void run() { if (mCirclePtImgv == null || mHandler == null) { return; } // 将中间圆圈View背景设置为开关打开状态然后开始向右平移 mCirclePtImgv.setBackgroundResource(R.drawable.switch_on_circle_point); startCirclePointAnim(); // 延迟100毫秒启动手指向下平移动画 mHandler.postDelayed(new Runnable() { @Override public void run() { // 手指向下移动开始时设置手指背景为正常的状态 if (mFingerImgv != null) { mFingerImgv.setBackgroundResource(R.drawable.finger_normal); } startFingerDownAnim(); } }, 100); } }, 200); } }); fingerUpAnim.start(); } /** * 手指向下移动动画 */ private void startFingerDownAnim() { if (mFingerImgv == null) { return; } ObjectAnimator fingerDownAnim = ObjectAnimator.ofFloat(mFingerImgv, "translationY", -mFingerMoveDistance, 0); fingerDownAnim.setDuration(FINGER_ANIM_DURATION); fingerDownAnim.addListener(new BaseAnimatorListener() { @Override public void onAnimationEnd(Animator animator) { // 手指向下移动动画完成,整个动画流程结束,重新开始下一次流程,循环执行动画,间隔1秒 mHandler.postDelayed(new Runnable() { @Override public void run() { if (isStopAnim) { return; } startAnim(); } }, 1000); } }); fingerDownAnim.start(); } }
最后一步:就是找个载体把SwitchOnAnimView加进去,调用其startAnim方法执行动画,这里在一个Activity中把播放此动画
定义activity布局文件activity_finger_switchon_anim.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/activity_animation_main" android:layout_width="match_parent" android:layout_height="match_parent" android:gravity="center" android:orientation="vertical"> <com.androidanimation.animationview.SwitchOnAnimView android:id="@+id/switch_on_anim_view" android:layout_width="wrap_content" android:layout_height="wrap_content"/> </LinearLayout>
定义并实现Activity:FingerSwitchOnAnimActivity
package com.androidanimation; import android.app.Activity; import android.os.Bundle; import android.os.Handler; import com.androidanimation.animationview.SwitchOnAnimView; public class FingerSwitchOnAnimActivity extends Activity { private Handler mHandler = new Handler(); private SwitchOnAnimView mSwitchOnAnimView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_finger_switchon_anim); mSwitchOnAnimView = (SwitchOnAnimView) findViewById(R.id.switch_on_anim_view); } @Override protected void onResume() { super.onResume(); mHandler.postDelayed(new Runnable() { @Override public void run() { mSwitchOnAnimView.startAnim(); } }, 500); } @Override protected void onPause() { super.onPause(); mSwitchOnAnimView.stopAnim(); } }
动画实现总结:
掌握Android的动画并不难,难的时候怎么实现一些复杂的动画,这里总结一下实现复杂动画的几个步骤。
1. 动画分解:任何复杂的动画都可以分解为很多个原子动画的组合
2. 动画衔接时机分析:复杂动画分解为很多个原子动画之后,要重新衔接起来
这里其实就是各个原子动画的执行时机,谁先谁后还是同时执行
3. 实现原子动画:将拆解的原子动画依次实现
4. 动画组装:上面都准备好之后,将原子动画按照一定的规律组装串联起来,整个复杂的动画就开始工作了
原子动画:本文指不能再继续拆分的动画
拿本文中的动画来说,动画可以分为四个:
a. 手指向上平移动画
b. 手指点击操作(这里不是动画,也可以当做一个简单的动画吧)
c. 开关按钮原点向右平移动画
d. 手指向下平移动画。
本文动画执行时机为:
a 先执行,a 执行完成之后立即执行 b,b 执行完成之后等待200ms执行 c(体现点击效果)
c 执行开始100ms后开始执行 d
动画的分解和动画衔接时机分析是不太容易的事,因为凭借肉眼有时候没法观察出来,所以播放动画的时候要放慢来看,如果还是不能看出来,最好还是要找公司的UI同事协助分析。因为我们能简单的区分平移动画,缩放动画这种简单,但是我们不能区分那种正弦算法动画或者是另外一些其他算法控制的动画。本文中的动画相对还是比较简单,实现起来也比较容易,但是思想确实一样的。
本文向大家介绍Android动画 实现开关按钮动画(属性动画之平移动画)实例代码,包括了Android动画 实现开关按钮动画(属性动画之平移动画)实例代码的使用技巧和注意事项,需要的朋友参考一下 Android动画 实现开关按钮动画(属性动画之平移动画),最近做项目,根据项目需求,有一个这样的功能,实现类似开关的动画效果,经过自己琢磨及上网查找资料,终于解决了,这里就记录下: 在Android
主要内容:本节引言:,1.Evaluator自定义,2.Interpolator(补间器),3.ViewPropertyAnimator,4.本节示例代码下载,本节小结本节引言: 上节我们对Android的属性动画进行了初步的学习,相信大家对于属性动画已经不再是 一知半解的状态了,本节我们继续来探究Android属性动画的一些更高级的用法! 依旧贴下郭神的三篇文章~ Android属性动画完全解析(上),初识属性动画的基本用法 Android属性动画完全解析(中),ValueAnimator和O
主要内容:本节引言:,1.属性动画概念叨叨逼,2.ValueAnimator简单使用,3.ObjectAnimator简单使用,4.组合动画与AnimatorListener,5.使用XML来编写动画,6.本节示例代码下载:,本节小结:本节引言: 本节给带来的是Android动画中的第三种动画——属性动画(Property Animation), 记得在上一节8.4.2 Android动画合集之补间动画为Fragment 设置过渡动画的时候,说过,App包和V4包下的Fragment调用setC
我正在使用一个语音识别应用程序,我想让我的播放/停止按钮在录制时“脉冲”。比如: 我曾试图使一个ScaleAnimation,做按钮增长,但当然,它使增长所有的按钮。 所以这个想法是实现类似的东西,但是只是在实际按钮后面有一个alpha。我想知道在我的按钮后面添加第二个“阿尔法按钮”以使它成长并达到这种效果之前,是否有可能用阿尔法动画或其他东西来做到这一点。
回头看看上一节显示的动画,它们是如何从一个空白的画布或白班变成一个复杂、漂亮的动画的?这些动画是怎么产生的?在变化过程中改变的特殊属性是什么?在回忆中摇晃你的手来表明一些物体应该移动或拉伸是不够的:如果你想要动画脱离你的手进入界面,你需要仔细思考在每一步中会发生什么以及哪些值被操作了。 如果你看一下上一节展示的动画GIF图,以及像CAPPTIVATE.co和其他网站展示的多种动画,并且你对发生了什
我正在制作一个鼓应用程序,我使用了键绑定,这样我就可以用键盘触发鼓的声音。 例如,我的的关键触发器是。我想做的是,当我按下时,在我的也会触发。所以实际上,看起来你真的在打鼓。 有什么想法可以让这成为可能吗?谢谢你们!:) 更多信息: 以下是我使用密钥绑定的方式: 我有一个名为KeyBind的类,它有6个返回类型Action的方法 在我的main表单中,我有6个方法也可以进行绑定。 然后我在Main