Android 动画系列一 视图动画(ViewAnimation

2018-09-11  本文已影响72人  赫丹

一、概述

Android3.0以前的原生动画
视图动画,也叫Tween(补间)动画可以在一个视图容器内执行一系列简单变换(位置、大小、旋转、透明度)。譬如,如果你有一个TextView对象,您可以移动、旋转、缩放、透明度设置其文本,当然,如果它有一个背景图像,背景图像会随着

二、相关属性详解

2-1 Animation(Animation抽象类是所有补间动画类的基类,基类中提供的一些通用的动画属性方法)

xml属性 java方法 说明
android:detachWallpaper setDetachWallpaper(boolean) 是否在壁纸上运行
android:duration setDuration(long) 动画持续时间,毫秒为单位
android:fillAfter setFillAfter(boolean) 控件动画结束时是否保持动画最后的状态
android:fillBefore setFillBefore(boolean) 控件动画结束时是否还原到开始动画前的状态
android:fillEnabled setFillEnabled(boolean) 与android:fillBefore效果相同
android:interpolator setInterpolator(Interpolator) 设定插值器(指定的动画效果,譬如回弹等
android:repeatCount setRepeatCount(int) 重复次数
android:repeatMode setRepeatMode(int) 重复类型有两个值,reverse表示倒序回放,restart表示从头播放
android:startOffset setStartOffset(long) 调用start函数之后等待开始运行的时间,单位为毫秒
android:zAdjustment setZAdjustment(int) 表示被设置动画的内容运行时在Z轴上的位置(top/bottom/normal),默认为normal
无论那种补间动画都具备以上的属性,可以是一个或多个,除此自外每种动画都有一些特有的属性;

2-2 AlpahAnimation(透明度动画)

xml属性 java方法 说明
android:fromAlpha AlphaAnimation(float fromAlpha, …) 动画开始的透明度(0.0到1.0,0.0是全透明,1.0是不透明)
android:toAlpha AlphaAnimation(…, float toAlpha) 动画结束的透明度(0.0到1.0,0.0是全透明,1.0是不透明)

2-3 RotateAnimation(旋转动画)

xml属性 java方法 说明
android:fromDegrees RotateAnimation(float fromDegrees, …) 旋转开始角度,正代表顺时针度数,负代表逆时针度数
android:toDegrees RotateAnimation(…, float toDegrees, …) 旋转结束角度,正代表顺时针度数,负代表逆时针度数
android:pivotX RotateAnimation(…, float pivotX, …) 缩放起点X坐标(数值、百分数、百分数p,譬如50表示以当前View左上角坐标加50px为初始点、50%表示以当前View的左上角加上当前View宽高的50%做为初始点、50%p表示以当前View的左上角加上父控件宽高的50%做为初始点)
android:pivotY RotateAnimation(…, float pivotY) 缩放起点Y坐标,同上规律

2-3 ScaleAnimation(缩放动画)

xml属性 java方法 说明
android:fromXScale ScaleAnimation(float fromX, …) 初始X轴缩放比例,1.0表示无变化
android:toXScale ScaleAnimation(…, float toX, …) 结束X轴缩放比例
android:fromYScale ScaleAnimation(…, float fromY, …) 初始Y轴缩放比例
android:toYScale ScaleAnimation(…, float toY, …) 结束Y轴缩放比例
android:pivotX ScaleAnimation(…, float pivotX, …) 缩放起点X轴坐标(数值、百分数、百分数p,譬如50表示以当前View左上角坐标加50px为初始点、50%表示以当前View的左上角加上当前View宽高的50%做为初始点、50%p表示以当前View的左上角加上父控件宽高的50%做为初始点)
android:pivotY ScaleAnimation(…, float pivotY) 缩放起点Y轴坐标,同上规律

2-4 TranslateAnimation(移动动画)

xml属性 java方法 说明
android:fromXDelta TranslateAnimation(float fromXDelta, …) 起始点X轴坐标(数值、百分数、百分数p,譬如50表示以当前View左上角坐标加50px为初始点、50%表示以当前View的左上角加上当前View宽高的50%做为初始点、50%p表示以当前View的左上角加上父控件宽高的50%做为初始点)
android:fromYDelta TranslateAnimation(…, float fromYDelta, …) 起始点Y轴从标,同上规律
android:toXDelta TranslateAnimation(…, float toXDelta, …) 结束点X轴坐标,同上规律
android:toYDelta TranslateAnimation(…, float toYDelta) 结束点Y轴坐标,同上规律

2-5 AnimationSet

AnimationSet继承自Animation,是上面四种的组合容器管理类,没有自己特有的属性,他的属性继承自Animation,所以特别注意,当我们对set标签使用Animation的属性时会对该标签下的所有子控件都产生影响。

三、相关方法详解

3-1 Animation类

方法 说明
reset() 重置Animation的初始化
cancel() 取消Animation动画
start() 开始Animation动画
setAnimationListener(AnimationListener listener) 给当前Animation设置动画监听
hasStarted() 判断当前Animation是否开始
hasEnded() 判断当前Animation是否结束

3-2 View类的常用动画操作方法

方法 说明
startAnimation(Animation animation) 对当前View开始设置的Animation动画
clearAnimation() 取消当View在执行的Animation动画

3-3 注意事项

补间动画执行之后并未改变View的真实布局属性值。切记这一点,譬如我们在Activity中有一个View在屏幕上方,我们设置了平移动画移动到屏幕下方然后保持动画最后执行状态呆在屏幕下方,这时如果点击屏幕下方动画执行之后的View是没有任何反应的,而点击原来屏幕上方没有Button的地方却响应的是点击View的事件。

四、使用方法

4-1 Alpha

方法1:在XML中设置
<?xml version="1.0" encoding="utf-8"?>
<alpha xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="3000"
    android:fillAfter="true"
    android:fromAlpha="1"
    android:repeatCount="5"
    android:repeatMode="reverse"
    android:startOffset="500"
    android:toAlpha="0"
    android:interpolator="@android:anim/overshoot_interpolator"
    ></alpha>
Animation  alphaAnimation = AnimationUtils.loadAnimation(this, R.anim.alpha);
 tween_img.startAnimation(alphaAnimation);
//取消动画
tween_img.clearAnimation();
方法2:在java代码中设置(效果同上)
AlphaAnimation  alphaAnimation = new AlphaAnimation(1, 0);
        alphaAnimation.setDuration(3000);
        alphaAnimation.setStartOffset(500);
        alphaAnimation.setFillAfter(false);
        alphaAnimation.setRepeatCount(5);
        alphaAnimation.setRepeatMode(Animation.RESTART);
        alphaAnimation.setInterpolator(new LinearInterpolator());
       tween_img.startAnimation(alphaAnimation);
      //取消动画
      tween_img.clearAnimation();

4-2 Rotate

方法1:在XML中设置
<?xml version="1.0" encoding="utf-8"?>
<rotate xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="3000"
    android:fillAfter="true"
    android:fillBefore="true"
    android:fromDegrees="0"
    android:interpolator="@android:anim/anticipate_interpolator"
    android:pivotX="50%"
    android:pivotY="50%"
    android:repeatCount="5"
    android:repeatMode="reverse"
    android:startOffset="500"
    android:toDegrees="180"
    ></rotate>
 RotateAnimation rotateAnimation = AnimationUtils.loadAnimation(this, R.anim.rotate); 
 tween_img.startAnimation(rotateAnimation);
//取消动画
tween_img.clearAnimation();
方法2:在java代码中设置(效果同上)
//        伸缩模式,可以取值为ABSOLUTE、RELATIVE_TO_SELF、RELATIVE_TO_PARENT
//        Animation.ABSOLUTE为绝对值 其他为百分比。
    RotateAnimation rotateAnimation = new RotateAnimation(0, 270, Animation.RELATIVE_TO_SELF, 0.5f, 
    Animation.RELATIVE_TO_SELF, 0.5f);
        rotateAnimation.setDuration(3000);
        rotateAnimation.setStartOffset(500);
        rotateAnimation.setFillAfter(false);
        rotateAnimation.setRepeatCount(5);
        rotateAnimation.setRepeatMode(Animation.REVERSE);
        rotateAnimation.setInterpolator(new LinearInterpolator());
       tween_img.startAnimation(rotateAnimation);
      //取消动画
      tween_img.clearAnimation();

4-3 Scale

方法1:在XML中设置
<?xml version="1.0" encoding="utf-8"?>
<scale xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="3000"
    android:fillAfter="true"
    android:fillBefore="true"
    android:interpolator="@android:anim/overshoot_interpolator"
    android:repeatCount="5"
    android:repeatMode="reverse"
    android:startOffset="500"

    android:fromXScale="1"
    android:fromYScale="1"
    android:toXScale="0"
    android:toYScale="0"
    android:pivotY="50%"
    android:pivotX="50%"
    ></scale>
 ScaleAnimation scaleAnimation = AnimationUtils.loadAnimation(this, R.anim.scale);
 tween_img.startAnimation(scaleAnimation);
//取消动画
tween_img.clearAnimation();
方法2:在java代码中设置(效果同上)
//        伸缩模式,可以取值为ABSOLUTE、RELATIVE_TO_SELF、RELATIVE_TO_PARENT
//        Animation.ABSOLUTE为绝对值 其他为百分比。
    AcaleAnimation scaleAnimation = new ScaleAnimation(1, 0, 1, 0, Animation.RELATIVE_TO_SELF, 0.5f, 
    Animation.RELATIVE_TO_SELF, 0.5f);
        scaleAnimation.setDuration(3000);
        scaleAnimation.setStartOffset(500);
        scaleAnimation.setFillAfter(false);
        scaleAnimation.setRepeatCount(5);
        scaleAnimation.setRepeatMode(Animation.REVERSE);
        scaleAnimation.setInterpolator(new LinearInterpolator());
       tween_img.startAnimation(scaleAnimation);
      //取消动画
      tween_img.clearAnimation();

4-4 Translate

方法1:在XML中设置
<?xml version="1.0" encoding="utf-8"?>
<translate xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="3000"
    android:fillAfter="true"
    android:fillBefore="true"
    android:interpolator="@android:anim/overshoot_interpolator"
    android:repeatCount="5"
    android:repeatMode="reverse"
    android:startOffset="500"

   android:fromXDelta="0"
    android:fromYDelta="0"
    android:toXDelta="50%p"
    android:toYDelta="0"
    ></translate>
 TranslateAnimation  translateAnimation = AnimationUtils.loadAnimation(this, R.anim.translate);
 tween_img.startAnimation(translateAnimation);
//取消动画
tween_img.clearAnimation();
方法2:在java代码中设置(效果同上)
//        伸缩模式,可以取值为ABSOLUTE、RELATIVE_TO_SELF、RELATIVE_TO_PARENT
//        Animation.ABSOLUTE为绝对值 其他为百分比。
    TranslateAnimation  translateAnimation = new TranslateAnimation(Animation.RELATIVE_TO_PARENT, -0.5f, Animation.RELATIVE_TO_PARENT, 0.5f,
                Animation.RELATIVE_TO_SELF, 0f,
                Animation.ABSOLUTE, 0);
        translateAnimation.setDuration(3000);
        translateAnimation.setStartOffset(500);
        translateAnimation.setFillAfter(true);
        translateAnimation.setFillAfter(false);
        translateAnimation.setRepeatCount(5);
        translateAnimation.setRepeatMode(Animation.REVERSE);
        translateAnimation.setInterpolator(new LinearInterpolator());
       tween_img.startAnimation(translateAnimation);
      //取消动画
      tween_img.clearAnimation();

4-5 Set

方法1:在XML中设置
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="500"
    android:fillAfter="true">
    <rotate
        android:duration="5000"
        android:fromDegrees="0"
        android:pivotX="50%"
        android:pivotY="50%"
        android:toDegrees="360" />
    <translate
        android:duration="5000"
        android:fromXDelta="-50%p"
        android:fromYDelta="0"
        android:toXDelta="50%p"
        android:toYDelta="0" />
    <alpha
        android:duration="5000"
        android:fromAlpha="1.0"
        android:toAlpha="0.0" />
    <scale
        android:duration="5000"
        android:fromXScale="1.0"
        android:fromYScale="1.0"
        android:pivotX="50%"
        android:pivotY="50%"
        android:toXScale="0.5"
        android:toYScale="0.5" />
</set>
 AnimationSet  setAnimation = AnimationUtils.loadAnimation(this, R.anim.set);
 tween_img.startAnimation(setAnimation);
//取消动画
tween_img.clearAnimation();
方法2:在java代码中设置(效果同上)
 //true使用集合的插值器,,false使用动画自己的插值器
       AnimationSet animationSet = new AnimationSet(true);
        animationSet.addAnimation(translateAnimation);
        animationSet.addAnimation(alphaAnimation);
//        animationSet.addAnimation(rotateAnimation);
//        animationSet.addAnimation(scaleAnimation);
        animationSet.setDuration(1000);
        animationSet.setFillAfter(false);
       tween_img.startAnimation(animationSet);
      //取消动画
      tween_img.clearAnimation();

五、动画监听

 animation.setAnimationListener(new Animation.AnimationListener() {
            @Override
            public void onAnimationStart(Animation animation) {
                //动画开始回调
            }

            @Override
            public void onAnimationEnd(Animation animation) {
                //动画结束回调
            }

            @Override
            public void onAnimationRepeat(Animation animation) {
                //重复执行动画回调

            }
        });

六、插值器(Interpolator)

6-1 插值器
java类 xml id值 说明
AccelerateDecelerateInterpolator @android:anim/accelerate_decelerate_interpolator 动画始末速率较慢,中间加速
AccelerateInterpolator @android:anim/accelerate_interpolator 动画开始速率较慢,之后慢慢加速
AnticipateInterpolator @android:anim/anticipate_interpolator 先退后再加速前进
AnticipateOvershootInterpolator @android:anim/anticipate_overshoot_interpolator 先退后再加速前进,超出终点后再回终点
BounceInterpolator @android:anim/bounce_interpolator 动画结束时弹球效果
CycleInterpolator @android:anim/cycle_interpolator 循环播放速率改变为正弦曲线
DecelerateInterpolator @android:anim/decelerate_interpolator 动画开始快然后慢
LinearInterpolator @android:anim/linear_interpolator 动画匀速改变
OvershootInterpolator @android:anim/overshoot_interpolator 向前弹出一定值之后回到原来位置
插值器主要用于android动画中,用于改变动画变化的速率。Android自定义了很多种插值器,可以满足一般动画的需求。不过,当用户对自己的动画有很精细的要求,那就需要我们自己来定义动画的Interpolator了
6-2 自定义插值器
一、 基于三次方贝塞尔曲线的插值器
1、先使用贝塞尔曲线数值生成工具来获取想要的曲线数值
2、次方曲线插值器.(基于三次方贝塞尔曲线)
public class BezierInterpolator implements Interpolator {

    private final static int ACCURACY = 4096;
    private int mLastI = 0;
    private final PointF mControlP1 = new PointF();
    private final PointF mControlP2 = new PointF();

    @Override
    public float getInterpolation(float input) {
        float t = input;

        for (int i = mLastI; i < ACCURACY; i++) {
            t = 1.0f * i / ACCURACY;
            double x = cubicCurves(t, 0, mControlP1.x, mControlP2.x, 1);
            if (x >= input) {
                mLastI = i;
                break;
            }
        }
        double value = cubicCurves(t, 0, mControlP1.y, mControlP2.y, 1);
       /* if (value > 0.999d) {
            value = 1;
            mLastI = 0;
        }*/
        return (float) value;
    }

    public BezierInterpolator(float x1, float y1, float x2, float y2) {
        mControlP1.x = x1;
        mControlP1.y = y1;
        mControlP2.x = x2;
        mControlP2.y = y2;
    }

    //求三次贝塞尔曲线(四个控制点)一个点某个维度的值
    public double cubicCurves(double t, double value0, double value1, double value2, double value3) {
        double value;
        double u = 1 - t;
        double tt = t * t;
        double uu = u * u;
        double uuu = uu * u;
        double ttt = tt * t;

        value = uuu * value0;
        value += 3 * uu * t * value1;
        value += 3 * u * tt * value2;
        value += ttt * value3;
        return value;
    }
}
3、使用方法同上,只需把获取的曲线控制点传入即可
二、通过实现Interpolator类来实现
public class LinearInterpolator extends BaseInterpolator implements NativeInterpolatorFactory {

    public LinearInterpolator() {
    }

    public LinearInterpolator(Context context, AttributeSet attrs) {
    }

    public float getInterpolation(float input) {
        return input;
    }

    /** @hide */
    @Override
    public long createNativeInterpolator() {
        return NativeInterpolatorFactoryHelper.createLinearInterpolator();
    }
}
public class AccelerateDecelerateInterpolator extends BaseInterpolator
        implements NativeInterpolatorFactory {
    public AccelerateDecelerateInterpolator() {
    }

    @SuppressWarnings({"UnusedDeclaration"})
    public AccelerateDecelerateInterpolator(Context context, AttributeSet attrs) {
    }

    public float getInterpolation(float input) {
        return (float)(Math.cos((input + 1) * Math.PI) / 2.0f) + 0.5f;
    }

    /** @hide */
    @Override
    public long createNativeInterpolator() {
        return NativeInterpolatorFactoryHelper.createAccelerateDecelerateInterpolator();
    }

public class SpringInterpolator implements Interpolator {
    private static final float factor = 0.2f;
    private float mFactor;

    public SpringInterpolator(float factor) {
        this.mFactor = factor;
    }

    public SpringInterpolator() {
        this(factor);
    }

    @Override
    public float getInterpolation(float input) {
//        factor = 0.1
//        pow(2, -10 * x) * sin((x - factor / 4) * (2 * PI) / factor) + 1
        float out = (float) (Math.pow(2, -10 * input) * Math.sin((input - mFactor / 4) * (2 * Math.PI) / mFactor) + 1);
        if(mListener!=null){
            mListener.onInterpolationChange(input,out);
        }
        return out;
    }

    public interface onInterpolationChangeListener
    {
        void onInterpolationChange(float y, float t);
    }

    onInterpolationChangeListener mListener = null;

    public void setOnInterpolationListener(onInterpolationChangeListener listener){
        mListener = listener;
    }
}
到这里一个自定义的Interpolator就完成了,主要的难点就在公式计算了,分享一个可视化插值器的网站,其中内置了一些插值器公式,还可以查看动画演示效果
Interpolator:http://inloop.github.io/interpolator/

七、应用场景

7-1 标准的动画效果
7-2 特殊的应用场景
Activity启动动画(自定义左右滑动效果)
需要定义四个动画,两个activity的进入和退出动画
 <translate
        android:duration="500"
        android:fromXDelta="100%p"
        android:toXDelta="0%p" />
 <translate
        android:duration="500"
        android:fromXDelta="0%p"
        android:toXDelta="-100%p" />
<translate
        android:duration="500"
        android:fromXDelta="-100%p"
        android:toXDelta="0%p" />
<translate
        android:duration="500"
        android:fromXDelta="0%p"
        android:toXDelta="100%p"/>
使用方法

 startActivity(new Intent(TweenActivity.this, InterpolatorActivity.class));
  overridePendingTransition(R.anim.activity_start_in_animation,R.anim.activity_start_out_animation);

 @Override
    public void finish() {
        super.finish();
        overridePendingTransition(R.anim.activity_finish_in_animation, R.anim.activity_finish_out_animation);
    }
上一篇 下一篇

猜你喜欢

热点阅读