一、使用objectAnimator实现下图的效果(不会做gif图)

点击前:

点击后

方法介绍:

public static ObjectAnimator ofFloat(Object target, String propertyName, float... values)   

第一个参数用于指定这个动画要操作的是哪个控件
第二个参数用于指定这个动画要操作这个控件的哪个属性
第三个参数是可变长参数,这个就跟ValueAnimator中的可变长参数的意义一样了,就是指这个属性值是从哪变到哪。

activity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"tools:context=".MainActivity"android:gravity="bottom|center_horizontal"><Buttonandroid:id="@+id/menu"style="@style/MenuStyle"android:background="@mipmap/ic_launcher" /><Buttonandroid:id="@+id/item1"style="@style/MenuItemStyle"android:background="@mipmap/logo"android:visibility="gone" /><Buttonandroid:id="@+id/item2"style="@style/MenuItemStyle"android:background="@mipmap/logo"android:visibility="gone" /><Buttonandroid:id="@+id/item3"style="@style/MenuItemStyle"android:background="@mipmap/logo"android:visibility="gone" /><Buttonandroid:id="@+id/item4"style="@style/MenuItemStyle"android:background="@mipmap/logo"android:visibility="gone" /><Buttonandroid:id="@+id/item5"style="@style/MenuItemStyle"android:background="@mipmap/logo"android:visibility="gone" /></RelativeLayout>

MainActivity.java

public class MainActivity extends ActionBarActivity implements View.OnClickListener {private static final String TAG = "MainActivity";private Button mMenuButton;private Button mItemButton1;private Button mItemButton2;private Button mItemButton3;private Button mItemButton4;private Button mItemButton5;private boolean mIsMenuOpen = false;private int len = 200;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();}private void initView() {mMenuButton = (Button) findViewById(R.id.menu);mMenuButton.setOnClickListener(this);mItemButton1 = (Button) findViewById(R.id.item1);mItemButton1.setOnClickListener(this);mItemButton2 = (Button) findViewById(R.id.item2);mItemButton2.setOnClickListener(this);mItemButton3 = (Button) findViewById(R.id.item3);mItemButton3.setOnClickListener(this);mItemButton4 = (Button) findViewById(R.id.item4);mItemButton4.setOnClickListener(this);mItemButton5 = (Button) findViewById(R.id.item5);mItemButton5.setOnClickListener(this);}/*** 打开菜单的动画* @param view 执行动画的view* @param index view在动画序列中的顺序* @param total 动画序列的个数* @param radius 动画半径*/private void doAnimateOpen(View view, int index, int total, int radius) {if (view.getVisibility() != View.VISIBLE) {view.setVisibility(View.VISIBLE);}double degree = Math.PI * index / ((total - 1) ); // 计算每个button移动的角度int translationX = (int) (radius * Math.cos(degree)); // 计算每个button在x轴移动的距离int translationY = -(int) (radius * Math.sin(degree)); // 计算每个button在-y轴移动的距离Log.d(TAG, String.format("degree=%f, translationX=%d, translationY=%d",degree, translationX, translationY));AnimatorSet set = new AnimatorSet();//包含平移、缩放和透明度动画set.playTogether(// 移动ObjectAnimator.ofFloat(view, "translationX", 0, translationX), ObjectAnimator.ofFloat(view, "translationY", 0, translationY),// 缩放ObjectAnimator.ofFloat(view, "scaleX", 0f, 1f),ObjectAnimator.ofFloat(view, "scaleY", 0f, 1f),// 透明度ObjectAnimator.ofFloat(view, "alpha", 0f, 1));//动画周期为500msset.setDuration(1 * 500).start();}/*** 关闭菜单的动画* @param view 执行动画的view* @param index view在动画序列中的顺序* @param total 动画序列的个数* @param radius 动画半径*/private void doAnimateClose(final View view, int index, int total,int radius) {if (view.getVisibility() != View.VISIBLE) {view.setVisibility(View.VISIBLE);}double degree = Math.PI * index / ((total - 1));// 计算每个button移动的角度int translationX = (int) (radius * Math.cos(degree));// 计算每个button在x轴移动的距离int translationY = -(int) (radius * Math.sin(degree));// 计算每个button在-y轴移动的距离Log.d(TAG, String.format("degree=%f, translationX=%d, translationY=%d",degree, translationX, translationY));AnimatorSet set = new AnimatorSet();//包含平移、缩放和透明度动画set.playTogether(// 移动ObjectAnimator.ofFloat(view, "translationX", translationX, 0),ObjectAnimator.ofFloat(view, "translationY", translationY, 0),// 缩放ObjectAnimator.ofFloat(view, "scaleX", 1f, 0f),ObjectAnimator.ofFloat(view, "scaleY", 1f, 0f),// 透明度ObjectAnimator.ofFloat(view, "alpha", 1f, 0f));//为动画加上事件监听,当动画结束的时候,我们把当前view隐藏set.addListener(new Animator.AnimatorListener() {@Overridepublic void onAnimationStart(Animator animator) {}@Overridepublic void onAnimationRepeat(Animator animator) {}@Overridepublic void onAnimationEnd(Animator animator) {view.setVisibility(View.GONE);}@Overridepublic void onAnimationCancel(Animator animator) {}});set.setDuration(1 * 500).start();}@Overridepublic void onClick(View v) {if (v == mMenuButton) {if (!mIsMenuOpen) {mIsMenuOpen = true;doAnimateOpen(mItemButton1, 0, 5, len);doAnimateOpen(mItemButton2, 1, 5, len);doAnimateOpen(mItemButton3, 2, 5, len);doAnimateOpen(mItemButton4, 3, 5, len);doAnimateOpen(mItemButton5, 4, 5, len);} else {mIsMenuOpen = false;doAnimateClose(mItemButton1, 0, 5, len);doAnimateClose(mItemButton2, 1, 5, len);doAnimateClose(mItemButton3, 2, 5, len);doAnimateClose(mItemButton4, 3, 5, len);doAnimateClose(mItemButton5, 4, 5, len);}} else {Toast.makeText(this, "你点击了" + v, Toast.LENGTH_SHORT).show();}}
}

styles.xml

  <style name="MenuStyle"><item name="android:layout_width">50dp</item><item name="android:layout_height">50dp</item></style><style name="MenuItemStyle"><item name="android:layout_width">45dp</item><item name="android:layout_height">45dp</item></style>

二、ObjectAnimator动画原理


在这张图中,将ValueAnimator的动画流程与ObjectAnimator的动画流程做了个对比。

可以看到ObjectAnimator的动画流程中,也是首先通过加速器产生当前进度的百分比,然后再经过Evaluator生成对应百分比所对应的数字值。这两步与ValueAnimator是完全一样的,唯一不同的是最后一步,在ValueAnimator中,我们要通过添加监听器来监听当前数字值。而在ObjectAnimator中,则是先根据属性值拼装成对应的set函数的名字,比如这里的scaleY的拼装方法就是将属性的第一个字母强制大写后,与set拼接,所以就是setScaleY。然后通过反射找到对应控件的setScaleY(float scaleY)函数,将当前数字值做为setScaleY(float scale)的参数将其传入。

这里在找到控件的set函数以后,是通过反射来调用这个函数的。
这就是ObjectAnimator的流程,最后一步总结起来就是调用对应属性的set方法,将动画当前数字值做为参数传进去。

根据上面的流程,这里有几个注意事项:
(1)、拼接set函数的方法:上面我们也说了是首先是强制将属性的第一个字母大写,然后与set拼接,就是对应的set函数的名字。注意,只是强制将属性的第一个字母大写,后面的部分是保持不变的。反过来,如果我们的函数名命名为setScalePointX(float ),那我们在写属性时可以写成“scalePointX”或者写成“ScalePointX”都是可以的,即第一个字母大小写可以随意,但后面的部分必须与set方法后的大小写保持一致。

(2)、如何确定函数的参数类型:上面我们知道了如何找到对应的函数名,那对应的参数方法的参数类型如何确定呢?我们在讲ValueAnimator的时候说过,动画过程中产生的数字值与构造时传入的值类型是一样的。由于ObjectAnimator与ValueAnimator在插值器和Evaluator这两步是完全一样的,而当前动画数值的产生是在Evaluator这一步产生的,所以ObjectAnimator的动画中产生的数值类型也是与构造时的类型一样的。那么问题来了,像我们的构造方法。

ObjectAnimator animator = ObjectAnimator.ofFloat(tv, "scaleY", 0, 3, 1);  

由于构造时使用的是ofFloat函数,所以中间值的类型应该是Float类型的,所以在最后一步拼装出来的set函数应该是setScaleY(float xxx)的样式;这时,系统就会利用反射来找到setScaleY(float xxx)函数,并把当前的动画数值做为参数传进去。
那问题来了,如果没有类似setScaleY(float xxx)的函数,我们只实现了一个setScaleY(int xxx)的函数怎么办?这里虽然函数名一样,但参数类型是不一样的,那么系统就会报一个错误:

意思就是对应函数的指定参数类型没有找到。

(3)、调用set函数以后怎么办?从ObjectAnimator的流程可以看到,ObjectAnimator只负责把动画过程中的数值传到对应属性的set函数中就结束了,注意传给set函数以后就结束了!set函数就相当我们在ValueAnimator中添加的监听的作用,set函数中的对控件的操作还是需要我们自己来写的。

那我们来看看View中的setScaleY是怎么实现的吧:

/** * Sets the amount that the view is scaled in Y around the pivot point, as a proportion of * the view's unscaled width. A value of 1 means that no scaling is applied. * * @param scaleY The scaling factor. * @see #getPivotX() * @see #getPivotY() * * @attr ref android.R.styleable#View_scaleY */
public void setScaleY(float scaleY) {  ensureTransformationInfo();  final TransformationInfo info = mTransformationInfo;  if (info.mScaleY != scaleY) {  invalidateParentCaches();  // Double-invalidation is necessary to capture view's old and new areas  invalidate(false);  info.mScaleY = scaleY;  info.mMatrixDirty = true;  mPrivateFlags |= DRAWN; // force another invalidation with the new orientation  invalidate(false);  }
} 

大家不必理解这一坨代码的意义,因为这些代码是需要读懂View的整体流程以后才能看得懂的,只需要跟着我的步骤来理解就行。这段代码总共分为两部分:第一步重新设置当前控件的参数,第二步调用Invalidate()强制重绘;
所以在重绘时,控件就会根据最新的控件参数来绘制了,所以我们就看到当前控件被缩放了。

(4)、set函数调用频率是多少:由于我们知道动画在进行时,每隔十几毫秒会刷新一次,所以我们的set函数也会每隔十几毫秒会被调用一次。

讲了这么多,就是为了强调一点:ObjectAnimator只负责把当前运动动画的数值传给set函数。至于set函数里面怎么来做,是我们自己的事了。

好了,在知道了ObjectAnimator的原理以后,下面就来看看如何自定义一个ObjectAnimator的属性吧。


三、自定义ObjectAnimator属性

上面我们已经看了使用View自带的set函数所对应属性的方法,而且理解了ObjectAnimator的动画实现原理,下面我们来自定义一个属性来看看实现效果吧。

我们在开始之前再来捋一下ObjectAnimator的动画设置流程:ObjectAnimator需要指定操作的控件对象,在开始动画时,到控件类中去寻找设置属性所对应的set函数,然后把动画中间值做为参数传给这个set函数并执行它。
所以,我们说了,控件类中必须所要设置属性所要对应的set函数。所以为了自由控制控件的实现,我们这里自定义一个控件。大家知道在这个自定义控件中,肯定存在一个set函数与我们自定义的属性相对应。
我们先来看看这段要实现的效果:

1、保存圆形信息类——Point
为了保存圆形的信息,我们先定义一个类:(Point.java)

public class Point {  private int mRadius;  public Point(int radius){  mRadius = radius;  }  public int getRadius() {  return mRadius;  }  public void setRadius(int radius) {  mRadius = radius;  }
}  

这个类很好理解,只有一个成员变量mRadius,表示圆的半径。

2、自定义控件——MyPointView
然后我们自定义一个控件MyPointView,完整代码如下:

public class MyPointView extends View {  private Point mPoint = new Point(100);  public MyPointView(Context context, AttributeSet attrs) {  super(context, attrs);  }  @Override  protected void onDraw(Canvas canvas) {  if (mPoint != null){  Paint paint = new Paint();  paint.setAntiAlias(true);  paint.setColor(Color.RED);  paint.setStyle(Paint.Style.FILL);  canvas.drawCircle(300,300,mPoint.getRadius(),paint);  }  super.onDraw(canvas);  }  void setPointRadius(int radius){  mPoint.setRadius(radius);  invalidate();  }  } 

在这段代码中,首先来看我们前面讲到的set函数:

void setPointRadius(int radius){  mPoint.setRadius(radius);  invalidate();  } 
  • 这个set函数所对应的属性应该是pointRadius或者PointRadius。前面我们已经讲了第一个字母大小写无所谓,后面的字母必须保持与set函数完全一致。

  • 在setPointRadius中,先将当前动画传过来的值保存到mPoint中,做为当前圆形的半径。然后强制界面刷新在界面刷新后,就开始执行onDraw()函数。

3、使用MyPointView
首先,在MyActivity的布局中添加MyPointView的使用(main.xml):

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  android:orientation="vertical"  android:layout_width="fill_parent"  android:layout_height="fill_parent">  <Button  android:id="@+id/btn"  android:layout_width="wrap_content"  android:layout_height="wrap_content"  android:layout_alignParentLeft="true"  android:padding="10dp"  android:text="start anim"  />  <Button  android:id="@+id/btn_cancel"  android:layout_width="wrap_content"  android:layout_height="wrap_content"  android:layout_alignParentRight="true"  android:padding="10dp"  android:text="cancel anim"  />  <TextView  android:id="@+id/tv"  android:layout_width="100dp"  android:layout_height="wrap_content"  android:layout_centerHorizontal="true"  android:gravity="center"  android:padding="10dp"  android:background="#ffff00"  android:text="Hello qijian"/>  <com.example.BlogObjectAnimator1.MyPointView  android:id="@+id/pointview"  android:layout_width="match_parent"  android:layout_height="match_parent"  android:layout_below="@id/tv"/>  </RelativeLayout>  

布局代码很好理解,根据效果图中的布局效果来理解,非常容易,就不再多讲
然后看看在MyActivity中,点击start anim后的处理方法:

public class MyActivity extends Activity {  private Button btnStart;  private MyPointView mPointView;  @Override  public void onCreate(Bundle savedInstanceState) {  super.onCreate(savedInstanceState);  setContentView(R.layout.main);  btnStart = (Button) findViewById(R.id.btn);  mPointView = (MyPointView)findViewById(R.id.pointview);  btnStart.setOnClickListener(new View.OnClickListener() {  @Override  public void onClick(View v) {  doPointViewAnimation();  }  });  }  …………
} 

在点击start anim按钮后,开始执行doPointViewAnimation()函数,doPointViewAnimation()函数代码如下:

private void doPointViewAnimation(){  ObjectAnimator animator = ObjectAnimator.ofInt(mPointView, "pointRadius", 0, 300, 100);  animator.setDuration(2000);  animator.start();
}  

在这段代码中,着重看ObjectAnimator的构造方法,首先要操作的控件对象是mPointView,然后对应的属性是pointRadius,然后值是从0到300再到100;
所以在动画开始以后,ObjectAnimator就会实时地把动画中产生的值做为参数传给MyPointView类中的setPointRadius(int radius)函数,然后调用setPointRadius(int radius)。由于我们在setPointRadius(int radius)中实时地设置圆形的半径值然后强制重绘当前界面,所以可以看到圆形的半径会随着动画的进行而改变。

四、注意——何时需要实现对应属性的get函数
我们再来看一下ObjectAinimator的下面三个构造方法:

public static ObjectAnimator ofFloat(Object target, String propertyName, float... values)
public static ObjectAnimator ofInt(Object target, String propertyName, int... values)
public static ObjectAnimator ofObject(Object target, String propertyName,TypeEvaluator evaluator, Object... values)  

前面我们已经分别讲过三个函数的使用方法,在上面的三个构造方法中最后一个参数都是可变长参数。我们也讲了,他们的意义就是从哪个值变到哪个值的。
那么问题来了:前面我们都是定义多个值,即至少两个值之间的变化,那如果我们只定义一个值呢,如下面的方式:(同样以MyPointView为例)

ObjectAnimator animator = ObjectAnimator.ofInt(mPointView, "pointRadius",100);  

从效果图中看起来是从0开始的,但是看log可以看出来已经在出警告了:

我们点了三次start anim按钮,所以这里也报了三次,意思就是没找到pointRadius属性所对应的getPointRadius()函数;

仅且仅当我们只给动画设置一个值时,程序才会调用属性对应的get函数来得到动画初始值。如果动画没有初始值,那么就会使用系统默认值。比如ofInt()中使用的参数类型是int类型的,而系统的Int值的默认值是0,所以动画就会从0运动到100;也就是系统虽然在找到不到属性对应的get函数时,会给出警告,但同时会用系统默认值做为动画初始值。

如果通过给自定义控件MyPointView设置了get函数,那么将会以get函数的返回值做为初始值:

public class MyPointView extends View {  private Point mPoint = new Point(100);  public MyPointView(Context context, AttributeSet attrs) {  super(context, attrs);  }  @Override  protected void onDraw(Canvas canvas) {  if (mPoint != null){  Paint paint = new Paint();  paint.setAntiAlias(true);  paint.setColor(Color.RED);  paint.setStyle(Paint.Style.FILL);  canvas.drawCircle(300,300,mPoint.getRadius(),paint);  }  super.onDraw(canvas);  }  public int getPointRadius(){  return 50;  }  public void setPointRadius(int radius){  mPoint.setRadius(radius);  invalidate();  }  } 

我们在这里添加了getPointRadius函数,返回值是Int.有些同学可能会疑惑:我怎么知道这里要返回int值呢?

我们前面说过当且仅当我们在创建ObjectAnimator时,只给他传递了一个过渡值的时候,系统才会调用属性对应的get函数来得到动画的初始值!所以做为动画的初始值,那么在创建动画时过渡值传的什么类型,这里的get函数就要返回类型

public static ObjectAnimator ofObject(Object target, String propertyName,TypeEvaluator evaluator, Object... values)  

比如上面的ofObject,get函数所返回的类型就是与最后一个参数Object… values,相同类型的。
在我们在MyPointView添加上PointRadius所对应的get函数以后重新执行动画:

ObjectAnimator animator = ObjectAnimator.ofInt(mPointView, "pointRadius",100);
animator.setDuration(2000);
animator.start();  


从动画中可以看出,半径已经不是从0开始的了,而是从50开始的。

最后我们总结一下:当且仅当动画的只有一个过渡值时,系统才会调用对应属性的get函数来得到动画的初始值。

参考内容:
http://blog.csdn.net/harvic880925/article/details/50598322
http://blog.csdn.net/singwhatiwanna/article/details/17639987

android动画---ObjectAnimator基本使用相关推荐

  1. Android 属性动画ObjectAnimator使用demo,组合动画

    //第一个参数:指定执行动画的控件,第二个参数:指定控件的属性,第三个参数是可变长参数 public static ObjectAnimator ofFloat(Object target, Stri ...

  2. 动画代码Android动画学习笔记动画代码

    间时紧张,先记一笔,后续优化与完善. 3.0之前,android支撑两种动画模式,tween animation,frame animation,在android3.0中又引入了一个新的动画统系:pr ...

  3. Android动画效果-更新中

    概述 Android系统提供了三种实现动画的方式,一种是补间动画(Tween Animation 在SDK中成为View Animation),另一种是帧动画(Frame Animation 在SDK ...

  4. Android 动画机制与使用技巧

    动画效果一直是人机交互中非常重要的部分,与死板.突兀的显示效果不同,动画效果的加入,让交互变得更加友好,特别是在提示.引导类的场景中,合理地使用动画能让用户获得更加愉悦的使用体验 一.Android ...

  5. android+动画队列,Android属性动画详解

    前言 属性动画是Android 3.0(API 11)新加入的动画框架,属性动画弥补了视图动画的很多短板,因此已经成为大多数动画场景的首选框架. 目录 目录 1. 属性动画出现的原因 在属性动画出现以 ...

  6. Android 动画(三)--属性动画

    Android Developer URL:http://developer.android.com/guide/topics/graphics/prop-animation.html 详细细节请参考 ...

  7. android动画详解

    转自:工匠若水 http://blog.csdn.net/yanbober 1 背景 不能只分析源码呀,分析的同时也要整理归纳基础知识,刚好有人微博私信让全面说说Android的动画,所以今天来一发A ...

  8. android动画文档,Android 动画系统汇总

    Android动画系统的种类: 1. 属性动画 (Property Animation) 2. 补间动画 (Tween Animation) 3. 帧动画     (Frame Animation) ...

  9. android动画入门,Android动画之入门篇(一)

    作为Android开发者,动画是非常重要的知识点,本文主要从入门角度来探索动画. Android的动画主要包括三大类:逐帧(Frame)动画,补间(Tween)动画,属性动画. 1. 逐帧(Frame ...

最新文章

  1. html显示假的图片路径,实现自己网站的图片假水印功能
  2. CASREL:A Novel Cascade Binary Tagging Framework for Relational Triple Extraction(关系抽取,ACL2020,重叠关系)
  3. linux杀掉80端口线程命令
  4. InnoSQL/MySQL并行复制的实现与配置
  5. python根据年月日计算天数_「每日一练」Python实现输入年月日计算第几天
  6. 95-190-028-源码-window-Window介绍与使用md
  7. Ubuntu系统下通过命令查找文件或文件夹
  8. 小心编译器的隐式声明
  9. windows7下bcdedit出现“拒绝访问”解决办法
  10. C#取得指定路径下所有目录及文件名称(可递归)
  11. 各种排序算法稳定性的探讨
  12. 制作CDKEY:CDKEY不宜包含生效时间
  13. ServerVariables 变量
  14. 皮尔逊相关系数和斯皮尔曼相关系数(等级系数)与典型相关分析
  15. 大学计算机基础vfp程序设计课程试验报告簿,VF程序设计实验报告册(实践教程).doc...
  16. 了解传销系列之三 : 开心门
  17. 制作一个html网页的步骤,制作一个完整的网页的步骤
  18. 任天堂游戏 html5,明年的预备阵容!任天堂承诺却还没出的作品
  19. 读书笔记 -《一生的计划》
  20. Android ImageView: resolveUri failed on bad bitmap uri

热门文章

  1. 鸿蒙系统p50什么时候上市,华为p50最新官方消息_华为p50什么时候上市
  2. 博士师兄给我推荐了两门课
  3. 《基于历史拥堵图和共识日识别的交通拥堵和出行时间预测》
  4. 虚拟机可以上网宿主机不能上网的解决情况
  5. html图片自适应表格,如何用纯CSS实现自适应布局表格
  6. vue中设置背景图片的方式
  7. Kobolds and Catacombs
  8. Java: 断言(assert)
  9. {*zoom:1} 作用
  10. X10服务器主板装系统黑屏,UEFI装系统失败黑屏的原因分析