Android中动画与属性动画

1、动画(Animation)

传统动画中,有位移、旋转、缩放、透明度变化等动画,如以下代码移动一个图片控件


TranslateAnimation animation = new TranslateAnimation(0, 200, 0, 0);
animation.setDuration(1000);
animation.setFillAfter(true);//使控件停留到动画结束的位置,若不设置则返回原来位置
iv.startAnimation(animation);

但iv的属性却留在了原地。iv的事件响应只在原来位置生效。而属性动画这可以实现属性随控件移动,如下句:
ObjectAnimator.ofFloat(iv, "translationX", 0, 200).setDuration(1000).start();
这句与以上代码会有相同的动画效果,且iv的响应事件也会跟随控件移动。

translationX或者translationY都是指控件的偏移量,也可以使用X,Y 指的是控件位置变化,如下句:
ObjectAnimator.ofFloat(iv, "Y", 0, 200).setDuration(1000).start();
这个方法的第二个参数是指控件中有get,set方法的属性名。只要有get与set方法就可以使用这种方法操作。如下句:

ObjectAnimator.ofFloat(iv, "rotation", 0, 360).setDuration(1000).start();
翻转360度


而若这三个动画放到一起,如下:

ObjectAnimator.ofFloat(iv, "rotation", 0, 360).setDuration(1000).start();
ObjectAnimator.ofFloat(iv, "Y", 0, 200).setDuration(1000).start();
ObjectAnimator.ofFloat(iv, "translationX", 0, 200).setDuration(1000).start();

则图片将会同时执行这三个动画,而不会执行一个后在执行另一个。这种效果也可以用以下方式;


PropertyValuesHolder p1=PropertyValuesHolder.ofFloat("rotation", 0, 360);
PropertyValuesHolder p2=PropertyValuesHolder.ofFloat( "Y", 0, 200);
PropertyValuesHolder p3=PropertyValuesHolder.ofFloat("translationX", 0, 200);

ObjectAnimator.ofPropertyValuesHolder(iv,p1,p2,p3).setDuration(1000).start();

以下也同样可以做到
ObjectAnimator a1 = ObjectAnimator.ofFloat(iv, "rotation", 0, 360);
ObjectAnimator a2 = ObjectAnimator.ofFloat(iv, "Y", 0, 200);
ObjectAnimator a3 = ObjectAnimator.ofFloat(iv, "translationX", 0, 200);

AnimatorSet set = new AnimatorSet();
set.playTogether(a1,a2,a3);
set.setDuration(1000);
set.start();


以上三种方法都是三个动画同时执行,若要按照顺序执行,这应该为:

ObjectAnimator a1 = ObjectAnimator.ofFloat(iv, "rotation", 0, 360);
ObjectAnimator a2 = ObjectAnimator.ofFloat(iv, "Y", 0, 200);
ObjectAnimator a3 = ObjectAnimator.ofFloat(iv, "translationX", 0, 200);

AnimatorSet set = new AnimatorSet();
set.playSequentially(a1,a2,a3);
set.setDuration(1000);
set.start();
当然也可以指定动画配合,设置几个同时执行,或者执行一个后在执行另外两个,如以下:
ObjectAnimator a1 = ObjectAnimator.ofFloat(iv, "rotation", 0, 360);
ObjectAnimator a2 = ObjectAnimator.ofFloat(iv, "Y", 0, 200);
ObjectAnimator a3 = ObjectAnimator.ofFloat(iv, "translationX", 0, 200);
AnimatorSet set = new AnimatorSet();
set.play(a1).with(a2).before(a3);
set.setDuration(1000);
set.start();


导致的效果就是,先旋转并向下,之后在向右移动


添加属性动画的监听事件:
ObjectAnimator a4 = ObjectAnimator.ofFloat(iv, "alpha", 0, 1);

a4.addListener(new Animator.AnimatorListener() {
@Override
public void onAnimationStart(Animator animation) {
LLogUtils.log();
}

@Override
public void onAnimationEnd(Animator animation) {
LLogUtils.log();
ToastUtil.show("结束了!!");
}

@Override
public void onAnimationCancel(Animator animation) {
LLogUtils.log();
}

@Override
public void onAnimationRepeat(Animator animation) {
LLogUtils.log();
}
});

a4.setDuration(1000);
a4.start();

或者使用另一种无需全部实现的listener

ObjectAnimator a4 = ObjectAnimator.ofFloat(iv, "alpha", 0, 1);

a4.addListener(new AnimatorListenerAdapter() {
@Override
public void onAnimationEnd(Animator animation) {
ToastUtil.show("结束了");
LLogUtils.log();
}
});

a4.setDuration(1000);
a4.start();



这样,可以自己选择要处理的事件并重写指定的方法,一般是结束时的事件。

下面,是一个实例,展示多个如元优酷APP的菜单的展开和回收,先看效果图。



技术分享技术分享技术分享技术分享
就是这样一个效果。
布局文件:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:padding="20dp"
  4. android:layout_width="match_parent"
  5. android:layout_height="match_parent">
  6. <ImageView
  7. android:src="@drawable/h"
  8. android:id="@+id/iv1"
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content"/>
  11. <ImageView
  12. android:src="@drawable/g"
  13. android:id="@+id/iv2"
  14. android:layout_width="wrap_content"
  15. android:layout_height="wrap_content"/>
  16. <ImageView
  17. android:id="@+id/iv3"
  18. android:src="@drawable/f"
  19. android:layout_width="wrap_content"
  20. android:layout_height="wrap_content"/>
  21. <ImageView
  22. android:src="@drawable/e"
  23. android:id="@+id/iv4"
  24. android:layout_width="wrap_content"
  25. android:layout_height="wrap_content"/>
  26. <ImageView
  27. android:src="@drawable/d"
  28. android:id="@+id/iv5"
  29. android:layout_width="wrap_content"
  30. android:layout_height="wrap_content"/>
  31. <ImageView
  32. android:src="@drawable/c"
  33. android:id="@+id/iv6"
  34. android:layout_width="wrap_content"
  35. android:layout_height="wrap_content"/>
  36. <ImageView
  37. android:src="@drawable/b"
  38. android:id="@+id/iv7"
  39. android:layout_width="wrap_content"
  40. android:layout_height="wrap_content"/>
  41. <ImageView
  42. android:src="@drawable/a"
  43. android:id="@+id/iv8"
  44. android:layout_width="wrap_content"
  45. android:layout_height="wrap_content"/>
  46. </FrameLayout>
代码文件:
package com.linwoain.test;

import android.animation.ObjectAnimator;
import android.os.Bundle;
import android.view.View;
import android.view.animation.AccelerateInterpolator;
import android.view.animation.BounceInterpolator;
import android.widget.ImageView;
import com.linwoain.ui.LinActivity;
import com.linwoain.util.LLogUtils;
import com.linwoain.util.ToastUtil;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends LinActivity implements View.OnClickListener {


private int[] res = {R.id.iv8, R.id.iv1, R.id.iv2, R.id.iv3, R.id.iv4, R.id.iv5, R.id.iv6, R.id.iv7};
private List<ImageView> imageViews = new ArrayList<ImageView>();
private boolean flag=true;


/**
* Called when the activity is first created.
*/
@Override

public void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);
setContentView(R.layout.main);
LLogUtils.log();

for (int i = 0; i < res.length; i++) {
ImageView iv = (ImageView) findViewById(res[i]);
iv.setOnClickListener(this);
imageViews.add(iv);
}


}

@Override
public void onClick(View v) {

switch (v.getId()) {

case R.id.iv8:
if (flag) {
startAnim();
} else {
closeAnim();
}
break;
default:
ToastUtil.show(v.hashCode()+"");
break;
}

}

private void closeAnim() {

for (int i = 1; i < res.length; i++) {
ObjectAnimator animator = ObjectAnimator.ofFloat(imageViews.get(i), "translationY", 110 * i, 0);
animator.setDuration(500);
animator.setStartDelay(i * 300);
animator.start();
animator.setInterpolator(new AccelerateInterpolator());
}
flag = true;

}

private void startAnim() {

for (int i = 1; i < res.length; i++) {
ObjectAnimator animator = ObjectAnimator.ofFloat(imageViews.get(i), "translationY", 0, 110 * i);
animator.setDuration(500);
animator.setStartDelay(i * 300);
animator.setInterpolator(new BounceInterpolator());//插值器
animator.start();
}
flag=false;
}
}

注意,动画中可以设置插值器,以实现特殊的效果,如动画运行中加速,减速,回弹等。可以参考apidemos中animation中的Interpolator













郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。