您现在的位置是:首页 > 数码 > 

android动画

2025-07-27 14:12:11
android动画 一、使用objectAnimator实现下图的效果(不会做gif图) 点击前: 点击后 方法介绍: public static ObjectAnimator ofFloat(Object target, String propertyame, float... values) 第一个参数用于指定

android动画

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

点击前:

点击后

方法介绍:

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

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

activity_main.xml

<RelativeLayout xmlns:android=xmlns:tools=android:layout_width=match_parentandroid:layout_height=match_parenttools:context=.MainActivityandroid:gravity=bottom|center_horizontal><Buttonandroid:id=@id/menustyle=@style/MenuStyleandroid:background=@mipmap/ic_launcher /><Buttonandroid:id=@id/item1style=@style/MenuItemStyleandroid:background=@mipmap/logoandroid:visibility=gone /><Buttonandroid:id=@id/item2style=@style/MenuItemStyleandroid:background=@mipmap/logoandroid:visibility=gone /><Buttonandroid:id=@id/itemstyle=@style/MenuItemStyleandroid:background=@mipmap/logoandroid:visibility=gone /><Buttonandroid:id=@id/item4style=@style/MenuItemStyleandroid:background=@mipmap/logoandroid:visibility=gone /><Buttonandroid:id=@id/item5style=@style/MenuItemStyleandroid:background=@mipmap/logoandroid: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 mItemButton;private Button mItemButton4;private Button mItemButton5;private boolean mIsMenuOpen = false;private int len = 200;@Overrideprotected void onCreate(Bundle savedInstanceState) {(savedInstanceState);setContentView(R.layout.activity_main);initView();}private void initView() {mMenuButton = (Button) findViewById(R.);mMenuButton.setOnClickListener(this);mItemButton1 = (Button) findViewById(R.id.item1);mItemButton1.setOnClickListener(this);mItemButton2 = (Button) findViewById(R.id.item2);mItemButton2.setOnClickListener(this);mItemButton = (Button) findViewById(R.id.item);mItemButton.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 * (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(// 移动(view, translationX, 0, translationX), (view, translationY, 0, translationY),// 缩放(view, scaleX, 0f, 1f),(view, scaleY, 0f, 1f),// 透明度(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 * (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(// 移动(view, translationX, translationX, 0),(view, translationY, translationY, 0),// 缩放(view, scaleX, 1f, 0f),(view, scaleY, 1f, 0f),// 透明度(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.GOE);}@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(mItemButton, 2, 5, len);doAnimateOpen(mItemButton4, , 5, len);doAnimateOpen(mItemButton5, 4, 5, len);} else {mIsMenuOpen = false;doAnimateClose(mItemButton1, 0, 5, len);doAnimateClose(mItemButton2, 1, 5, len);doAnimateClose(mItemButton, 2, 5, len);doAnimateClose(mItemButton4, , 5, len);doAnimateClose(mItemButton5, 4, 5, len);}} else {(this, 你点击了  v, Toast.LEGTH_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 = (tv, scaleY, 0, , 1);  

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

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

()、调用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 ( != scaleY) {  invalidateParentCaches();  // Double-invalidation is necessary to capture view	s old and new areas  invalidate(false);   = scaleY;   = true;  mPrivateFlags |= DRAW; // 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(00,00,mPoint.getRadius(),paint);  }  (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()函数。

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

<?xml version=1.0 encoding=utf-8?>  
<RelativeLayout xmlns: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/>  <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) {  (savedInstanceState);  setContentView(R.);  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 = (mPointView, pointRadius, 0, 00, 100);  animator.setDuration(2000);  animator.start();  
}  

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

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

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

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

ObjectAnimator animator = (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(00,00,mPoint.getRadius(),paint);  }  (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 propertyame,TypeEvaluator evaluator, Object... values)  

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

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


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

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

参考内容:

#感谢您对电脑配置推荐网 - 最新i3 i5 i7组装电脑配置单推荐报价格的认可,转载请说明来源于"电脑配置推荐网 - 最新i3 i5 i7组装电脑配置单推荐报价格

本文地址:http://www.dnpztj.cn/shuma/707473.html

相关标签:无
上传时间: 2023-11-21 21:06:16

上一篇:2022

下一篇:多个服务器数据互通

留言与评论(共有 5 条评论)
本站网友 sh906
2分钟前 发表
那我们来看看View中的setScaleY是怎么实现的吧: /** * Sets the amount that the view is scaled in Y around the pivot point
本站网友 孙东明
17分钟前 发表
len);doAnimateOpen(mItemButton
本站网友 大肚能容容天下难容之事
27分钟前 发表
5
本站网友 长春不孕不育医院
9分钟前 发表
int total