继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

Android动画解析(二)

哈士奇WWW
关注TA
已关注
手记 550
粉丝 71
获赞 402

一、插值器和估值器:

Interpolator(插值器)和TypeEvaluator(估值器)是实现非匀速动画的重要手段

1、Interpolator: 过渡值变化的规则( 控制动画的进度 )

  1. * AccelerateDecelerateInterpolator   在动画开始与介绍的地方速率改变比较慢,在中间的时候加速  

  2. * AccelerateInterpolator                     在动画开始的地方速率改变比较慢,然后开始加速  

  3. * AnticipateInterpolator                      开始的时候向后然后向前甩  

  4. * AnticipateOvershootInterpolator     开始的时候向后然后向前甩一定值后返回最后的值  

  5. * BounceInterpolator                          动画结束的时候弹起  

  6. * CycleInterpolator                             动画循环播放特定的次数,速率改变沿着正弦曲线  

  7. * DecelerateInterpolator                    在动画开始的地方快然后慢  

  8. * LinearInterpolator                            以常量速率改变  

  9. * OvershootInterpolator                      向前甩一定值后再回到原来位置  

2、自定义插值器:
实现TimeInterpolator。TimeInterpolator:时间插值器,根据时间的百分比计算属性改变的百分比


  1. class MyInterploator implements TimeInterpolator{  

  2.      @Override  

  3.      public float getInterpolation(float input) {  

  4.          return 1-input;  

  5.      }  

  6.  }  

在getInterpolation函数中,直接把input值返回,即以当前动画的进度做为动画的数值进度,这也就表示当前动画的数值进度与动画的时间进度一致

3、evaluator  估值器

将从加速器返回的数字进度转成对应的数字值。

通过在AnimatorUpdateListener监听器使用animation.getAnimatedValue()函数拿到Evaluator中返回的数字值。 

ofInt()对应的Evaluator类名叫IntEvaluator,而ofFloat()对应的Evaluator类名叫FloatEvaluator; 

  1. public class IntEvaluator implements TypeEvaluator<Integer> {  

  2.   

  3.     /** 

  4.      * This function returns the result of linearly interpolating the start and end values, with 

  5.      * <code>fraction</code> representing the proportion between the start and end values. The 

  6.      * calculation is a simple parametric calculation: <code>result = x0 + t * (v1 - v0)</code>, 

  7.      * where <code>x0</code> is <code>startValue</code>, <code>x1</code> is <code>endValue</code>, 

  8.      * and <code>t</code> is <code>fraction</code>. 

  9.      * 

  10.      * @param fraction   The fraction from the starting to the ending values 

  11.      * @param startValue The start value; should be of type <code>int</code> or 

  12.      *                   <code>Integer</code> 

  13.      * @param endValue   The end value; should be of type <code>int</code> or <code>Integer</code> 

  14.      * @return A linear interpolation between the start and end values, given the 

  15.      *         <code>fraction</code> parameter. 

  16.      */  

  17.     public Integer evaluate(float fraction, Integer startValue, Integer endValue) {  

  18.         int startInt = startValue;  

  19.         return (int)(startInt + fraction * (endValue - startInt));  

  20.     }  

  21. }  

fraction就是加速器中的返回值,表示当前动画的数值进度,百分制的小数表示。 

startValue和endValue分别对应ofInt(int start,int end)中的start和end的数值;

4、自定义估值器:

  1. public class MyEvaluator implements TypeEvaluator<Integer> {    

  2.        @Override    

  3.        public Integer evaluate(float fraction, Integer startValue, Integer endValue) {  

  4.            int startInt = startValue;  

  5.            return (int)(200+startInt + fraction * (endValue - startInt));  

  6.        }    

  7. }  

5、ArgbEvalutor  颜色过度的估值器

  1. ValueAnimator animator = ValueAnimator.ofInt(0xffffff000xff0000ff);  

  2. animator.setEvaluator(new ArgbEvaluator());  

  3. animator.setDuration(3000);  

  4.    animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener(){  

  5.        @Override  

  6.        public void onAnimationUpdate (ValueAnimator animation){  

  7.            int curValue = (int) animation.getAnimatedValue();  

  8.            tv.setBackgroundColor(curValue);  

  9.         }  

  10.    });  

  11.    animator.start();  

三、ObjectAnimator 

继承于ValueAnimator

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)

  • 第一个参数用于指定这个动画要操作的是哪个控件

  • 第二个参数用于指定这个动画要操作这个控件的哪个属性

  • 第三个参数是可变长参数,这个就跟ValueAnimator中的可变长参数的意义一样了,就是指这个属性值是从哪变到哪。

1、属性动画的原理:

object必须提供该属性的set方法,多次调用set方法,每次传给set的值不一样。

如果没初始值,还需要get方法,因为系统要获取属性的初始值。

例如:

  1. ObjectAnimator animator = ObjectAnimator.ofFloat(tv,"rotation",0,180,0);  

ObjectAnimator在做动画时就会到指定控件(TextView)中去找对应的setRotation()方法来改变控件中对应的值。

  1. public void setRotation(float rotation) {  

  2.     if (rotation != getRotation()) {  

  3.         // Double-invalidation is necessary to capture view's old and new areas  

  4.         invalidateViewProperty(truefalse);  

  5.         mRenderNode.setRotation(rotation);  

  6.         invalidateViewProperty(falsetrue);  

  7.   

  8.         invalidateParentIfNeededAndWasQuickRejected();  

  9.         notifySubtreeAccessibilityStateChangedIfNeeded();  

  10.     }  

  11. }  

调用的内部进行属性调整,通过invalidate进行重绘。


View关于动画的set方法如下:

//透明度:alpha 

public void setAlpha(float alpha) 

//旋转度数:rotation、rotationX、rotationY 

public void setRotation(float rotation) 

public void setRotationX(float rotationX) 

public void setRotationY(float rotationY) 

//平移:translationX、translationY 

public void setTranslationX(float translationX)   

public void setTranslationY(float translationY) 

//缩放:scaleX、scaleY 

public void setScaleX(float scaleX) 

public void setScaleY(float scaleY)

2、自定义控件


  1. public class MyPointView extends View {  

  2.     private Point mPoint = new Point(100);  

  3.   

  4.     public MyPointView(Context context, AttributeSet attrs) {  

  5.         super(context, attrs);  

  6.     }  

  7.   

  8.     @Override  

  9.     protected void onDraw(Canvas canvas) {  

  10.         if (mPoint != null){  

  11.             Paint paint = new Paint();  

  12.             paint.setAntiAlias(true);  

  13.             paint.setColor(Color.RED);  

  14.             paint.setStyle(Paint.Style.FILL);  

  15.             canvas.drawCircle(300,300,mPoint.getRadius(),paint);  

  16.         }  

  17.         super.onDraw(canvas);  

  18.     }  

  19.   

  20.     void setPointRadius(int radius){  

  21.         mPoint.setRadius(radius);  

  22.         invalidate();  

  23.     }  

  24.      //初始值  

  25.      public int getPointRadius(){  

  26.           return 20;  

  27.      }  

  28.   

  29. }   

  1. private void doPointViewAnimation(){  

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

  3.     animator.setDuration(2000);  

  4.     animator.start();  

  5. }   

四、PropertyValuesHolder

1、保存动画过程所需操作的属性和对应的值。

ofFloat内部实现是将传入的参数封装为PropertyValuesHolder 。

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

  2.     ObjectAnimator anim = new ObjectAnimator(target, propertyName);  

  3.     anim.setFloatValues(values);  

  4.     return anim;  

  5. }  

  1. @Override  

  2. public void setFloatValues(float... values) {  

  3.     if (mValues == null || mValues.length == 0) {  

  4.         // No values yet - this animator is being constructed piecemeal. Init the values with  

  5.         // whatever the current propertyName is  

  6.         if (mProperty != null) {  

  7.             setValues(PropertyValuesHolder.ofFloat(mProperty, values));  

  8.         } else {  

  9.             setValues(PropertyValuesHolder.ofFloat(mPropertyName, values));  

  10.         }  

  11.     } else {  

  12.         super.setFloatValues(values);  

  13.     }  

  14. }  

  1. public void setValues(PropertyValuesHolder... values) {  

  2.     int numValues = values.length;  

  3.     mValues = values;  

  4.     mValuesMap = new HashMap<String, PropertyValuesHolder>(numValues);  

  5.     for (int i = 0; i < numValues; ++i) {  

  6.         PropertyValuesHolder valuesHolder = values[i];  

  7.         mValuesMap.put(valuesHolder.getPropertyName(), valuesHolder);  

  8.     }  

  9.     // New property/values/target should cause re-initialization prior to starting  

  10.     mInitialized = false;  

  11. }  

2、构造PropertyValuesHolder:

  1. public static PropertyValuesHolder ofFloat(String propertyName, float... values)  

  2. public static PropertyValuesHolder ofInt(String propertyName, int... values)  

  3. public static PropertyValuesHolder ofObject(String propertyName, TypeEvaluator evaluator, Object... values)  

  4. public static PropertyValuesHolder ofKeyframe(String propertyName, Keyframe... values)  

propertyName: 需要操作的参数名

values : 属性对应可变参数

evaluator:估值器

3、返回ObjectAnimator对象

  1. ObjectAnimator.ofPropertyValuesHolder(Object target,PropertyValuesHolder... values)  

target: 需要执行动画的控件

values : 可以是多个PropertyValuesHolder,每一个PropertyValuesHolder对一个属性做动画,多个实例可对控件多个属性动画。

  1. PropertyValuesHolder rotationHolder = PropertyValuesHolder.ofFloat("Rotation", 60f, -60f, 40f, -40f, -20f, 20f, 10f, -10f, 0f);  

  2. PropertyValuesHolder colorHolder = PropertyValuesHolder.ofInt("BackgroundColor"0xffffffff0xffff00ff0xffffff000xffffffff);  

  3. ObjectAnimator animator = ObjectAnimator.ofPropertyValuesHolder(mTextView, rotationHolder, colorHolder);  

  4. animator.setDuration(3000);  

  5. animator.setInterpolator(new AccelerateInterpolator());  

  6. animator.start();  

类似于动画集合

(3)ofObject:

例:改变TextView textSize属性

注意:改变属性的类型要匹配,控件有set方法(可自己扩展)

  1. private void startAnimator() {  

  2.     PropertyValuesHolder charHolder = PropertyValuesHolder.ofObject("textSize",new FloatEvaluator(),10f,20f);  

  3.     ObjectAnimator animator = ObjectAnimator.ofPropertyValuesHolder(myText, charHolder);  

  4.     animator.setDuration(5000);  

  5.     animator.setInterpolator(new AccelerateInterpolator());  

  6.     animator.start();  

  7. }  

  8.   

  9. class FloatEvaluator implements TypeEvaluator<Float>{  

  10.     @Override  

  11.     public Float evaluate(float fraction, Float startValue, Float endValue) {  

  12.         return fraction * (endValue -startValue);  

  13.     }  

  14. }  

4、Keyframe: 关键帧,为了控制动画速率

  1. /** 

  2.  * ofFloat  

  3.  */  

  4. public static Keyframe ofFloat(float fraction)  

  5. public static Keyframe ofFloat(float fraction, float value)  

  6. /** 

  7.  * ofInt  

  8.  */  

  9. public static Keyframe ofInt(float fraction)  

  10. public static Keyframe ofInt(float fraction, int value)   

fraction:当前进度

value:当前位置

例:


  1. Keyframe frame0 = Keyframe.ofFloat(0f, 0);  

  2. Keyframe frame1 = Keyframe.ofFloat(0.1f, -20f);  

  3. Keyframe frame2 = Keyframe.ofFloat(10);  

  4. PropertyValuesHolder frameHolder = PropertyValuesHolder.ofKeyframe("rotation",frame0,frame1,frame2);  

  5. Animator animator = ObjectAnimator.ofPropertyValuesHolder(mImage,frameHolder);   

  6. animator.setDuration(1000);   

  7. animator.start();  

2)常用函数:

  1. /** 

  2.  * 设置fraction参数,即Keyframe所对应的进度  

  3.  */  

  4. public void setFraction(float fraction)  

  5. /** 

  6.  * 设置当前Keyframe所对应的值  

  7.  */  

  8. public void setValue(Object value)  

  9. /** 

  10.  * 前一个Keyframe到当前Keyframe期间所对应的插值器  

  11.  */  

  12. public void setInterpolator(TimeInterpolator interpolator)   

注:

  • 如果去掉第0帧,将以第一个关键帧为起始位置

  • 如果去掉结束帧,将以最后一个关键帧为结束位置

  • 使用Keyframe来构建动画,至少要有两个或两个以上帧

五、AnimatorSet 

1、 playSequentially表示所有动画依次播放,playTogether表示所有动画一起开始。

  1. public void playSequentially(Animator... items);  

  2. public void playSequentially(List<Animator> items);  

  3. public void playTogether(Animator... items);  

  4. public void playTogether(Collection<Animator> items);  

  1. private void doPlaySequentiallyAnimator(){  

  2.     ObjectAnimator tv1BgAnimator = ObjectAnimator.ofInt(mTv1, "BackgroundColor",  0xffff00ff0xffffff000xffff00ff);  

  3.     ObjectAnimator tv1TranslateY = ObjectAnimator.ofFloat(mTv1, "translationY"03000);  

  4.     ObjectAnimator tv2TranslateY = ObjectAnimator.ofFloat(mTv2, "translationY"04000);  

  5.   

  6.     AnimatorSet animatorSet = new AnimatorSet();  

  7.     animatorSet.playSequentially(tv1BgAnimator,tv1TranslateY,tv2TranslateY);  

  8.     animatorSet.setDuration(1000);  

  9.     animatorSet.start();  

  10. }  

注:

  • 如果去掉第0帧,将以第一个关键帧为起始位置

  • 如果去掉结束帧,将以最后一个关键帧为结束位置

  • 使用Keyframe来构建动画,至少要有两个或两个以上帧


2、AnimatorSet.Builder

  1. //表示要播放哪个动画  

  2. public Builder play(Animator anim)  

  3. //和前面动画一起执行  

  4. public Builder with(Animator anim)  

  5. //执行前面的动画后才执行该动画  

  6. public Builder before(Animator anim)  

  7. //执行先执行这个动画再执行前面动画  

  8. public Builder after(Animator anim)  

  9. //延迟n毫秒之后执行动画  

  10. public Builder after(long delay)  

串行方式执行:

  1. ObjectAnimator tv1BgAnimator = ObjectAnimator.ofInt(mTv1, "BackgroundColor",  0xffff00ff0xffffff000xffff00ff);  

  2. ObjectAnimator tv1TranslateY = ObjectAnimator.ofFloat(mTv1, "translationY"04000);  

  3. ObjectAnimator tv2TranslateY = ObjectAnimator.ofFloat(mTv2, "translationY"04000);  

  4.   

  5. AnimatorSet animatorSet = new AnimatorSet();  

  6. animatorSet.play(tv1TranslateY).with(tv2TranslateY).after(tv1BgAnimator);  

  7. animatorSet.setDuration(2000);  

  8. animatorSet.start();  

3、函数:

  1. //设置单次动画时长  

  2. public AnimatorSet setDuration(long duration);  

  3. //设置加速器  

  4. public void setInterpolator(TimeInterpolator interpolator)  

  5. //设置ObjectAnimator动画目标控件  

  6. public void setTarget(Object target)  

  7. //设置延时开始动画时长(只是延长AnimatorSet的开始时间)  

  8. public void setStartDelay(long startDelay)  

四、对任意属性做动画(自定义动画):

1、属性动画的原理:

(1)object必须提供该属性的set方法,多次调用set方法,每次传给set的值不一样。

如果没初始值,还需要get方法,因为系统要获取属性的初始值。


(2)对于属性的改变必须通过某种ui的变化体现出来(不然动画无效)

解决方式:

1、给你的对象加上set和get方法(或者用一个类包装原始类,间接提供get和set方法。)

  1. public class Animator {  

  2.     private Button button;  

  3.     private void performAnimate(){  

  4.         //按钮点击后执行的动画  

  5.         ViewWrapper viewWrapper = new ViewWrapper(button);  

  6.         ObjectAnimator.ofInt(viewWrapper,"width",500).setDuration(5000).start();  

  7.     }  

  8.   

  9.     private static class ViewWrapper{  

  10.         private View mtagerView;  

  11.   

  12.         public ViewWrapper(View target){  

  13.             mtagerView = target;  

  14.         }  

  15.   

  16.         public int getWidth(){  

  17.             return mtagerView.getLayoutParams().width;  

  18.         }  

  19.         public void setWidth(int width){  

  20.             mtagerView.getLayoutParams().width = width;  

  21.             mtagerView.requestLayout();  

  22.         }  

  23.     }  

  24. }  

2、采用ValueAnimator,监听动画过程,自己实现属性的改变

  1. public void performAnimate(final int start, final int end, final View targetView){  

  2.     final ValueAnimator valueAnimator =ValueAnimator.ofInt(1,100);  

  3.     valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {  

  4.         //整形估值器,估值时使用  

  5.         private IntEvaluator intEvaluator = new IntEvaluator();  

  6.         @Override  

  7.         public void onAnimationUpdate(ValueAnimator animation) {  

  8.             //获取动画的进度  

  9.             int currValue=(Integer) animation.getAnimatedValue();  

  10.             //当前进度占整个动画的比例  

  11.             float fraction= animation.getAnimatedFraction();  

  12.             //通过比例计算出宽度,然后在设给Button  

  13.             targetView.getLayoutParams().width=intEvaluator.evaluate(fraction,start,end);  

  14.             targetView.requestLayout();  

  15.         }  

  16.     });  

  17.     valueAnimator.setDuration(5000).start(); //5000ms内1-100  

  18. }  

3、使用动画注意事项

  • OOM:数量较多并图片较大

  • 内存泄露:无限循环动画,需要在Activity中退出时停止

  • 兼容性问题

  • View动画完成,View无法隐藏。需要调用clearAnimation清除View动画

  • 尽量使用dp

  • 动画过程,开启硬件加速

原文链接:http://www.apkbus.com/blog-847095-77492.html

打开App,阅读手记
0人推荐
发表评论
随时随地看视频慕课网APP