3.ValueAnimator 属性动画

动画介绍:

  • 在Android动画中,总共有两种类型的动画View Animation(视图动画)和Property Animator(属性动画);
  • View Animation包括Tween Animation(补间动画)和Frame Animation(逐帧动画);
    Property Animator包括ValueAnimator和ObjectAnimation;

区别:

  • View Animation仅能对指定的控件做动画,而Property Animator是通过改变控件某一属性值来做动画的。
  • 补间动画虽能对控件做动画,但并没有改变控件内部的属性值。而Property Animator则是恰恰相反,Property Animator是通过改变控件内部的属性值来达到动画效果的。

属性动画:

ValueAnimator

  • 注意:单从名字上就可以看出来,这个Animator是针对数值进行操作的,不对控件做操作。控件和它并没有什么关系,重点在数值上的操作。
  • 两点:
    • ValueAnimator只负责对指定数值区间,进行运算。
    • 自己需要对运算进行监听,然后自己对控件进行操作。
ValueAnimator animator = ValueAnimator.ofInt(0,400);  
animator.setDuration(1000);  
animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {  
    @Override  
    public void onAnimationUpdate(ValueAnimator animation) {  
        //拿到监听结果,自己处理。
        int curValue = (int)animation.getAnimatedValue(); 
        tvTextView.layout(curValue,curValue,curValue+tv.getWidth(),curValue+tv.getHeight());
    }  
});  
animator.start();  
常用方法和监听:
  • 方法:和Animation差不多,看文档。
  • 监听:
    • AnimatorUpdateListener:监听动画变化时的实时值。addUpdateListener(AnimatorUpdateListener listener)。
    • AnimatorListener:监听动画变化时四个状态。addListener(AnimatorListener listener)。
  • 移除监听:
/** 
 * 移除AnimatorUpdateListener 
 */  
void removeUpdateListener(AnimatorUpdateListener listener);  
void removeAllUpdateListeners();  
 /** 
  * 移除AnimatorListener 
  */  
void removeListener(AnimatorListener listener);  
void removeAllListeners();  

查看这段代码:

ValueAnimator animator = ValueAnimator.ofInt(0,600); 
animator.setDuration(1000);  
animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {  
    @Override  
    public void onAnimationUpdate(ValueAnimator animation) {  
        int curValue = (int)animation.getAnimatedValue();  
    }  
});  
animator.setInterpolator(new LinearInterpolator());  
animator.start();  
上面代码中,在1000毫秒的时间中,ValueAnimator的功能是,使数值从0到600根据插值器的类型去变化。 (int)animation.getAnimatedValue()得到的是0~600之间的值,这个数值的获取是Interpolator和Evalutor作用的结果。
  • 0~600得到的值是否是匀速变化还是其他速度变化的,由Interpolator决定。
  • animation.getAnimatedValue()得到的值是多少,什么类型由Evalutor决定。
inInt(0,600) 加速器 估值器 监听器返回
设置数值区间 Interpolator动画比例 根据加速器的动画比例算出实际值 getAnimatedValue得到实际值

插值器/加速器 Interpolator

  • 默认每10毫秒刷新一次
    //线性加速器
public class LinearInterpolator implements Interpolator {  
    public LinearInterpolator() {}  
    public LinearInterpolator(Context context, AttributeSet attrs) {}  
    public float getInterpolation(float input) {  
        return input;  
    }  
}  
public interface Interpolator extends TimeInterpolator {}  

参数input:是一个float类型,它取值范围是0到1,表示当前动画的进度,取0时表示动画刚开始,取1时表示动画结束,取0.5时表示动画中间的位置,其它类推。
返回值:表示当前实际想要显示的进度。取值可以超过1也可以小于0,超过1表示已经超过目标值,小于0表示小于开始位置。

再重复一遍,input参数与任何我们设定的值没关系,只与时间有关,随着时间的增长,动画的进度也自然的增加,input参数就代表了当前动画的进度。而返回值则表示自定义后的动画的当前数值进度。

上面代码:为什么LinearInterpolator是线性匀速变化的呢?
因为插值器根据设置的总时间,每10ms刷新一次,调用getInterpolation传入当前的进度,LinearInterpolator中原值返回了,没有改变默认进度。

看自定义加速器代码对比:
public class MyInterpolator implements Interpolator {
    /**
     * input 是实际动画执行的时间比例 0~1
     * newInput 你想让动画已经执行的比例 0~1。
     * 注意:都是比例,而不是实际的值。
     * 
     * setDuration(1000)情况下:前200ms走了3/4的路程比例,后800ms走了1/4的路程比例。
     * 注意:这里输出的是比例,是路程比例,不是实际的路程值!
     */
    @Override
    public float getInterpolation(float input) {
        if (input <= 0.2) {//后1/4的时间,输出3/4的比例
            float newInput = input*4;
            return newInput;
        }else {//后3/4的时间,输出1/4的比例
            float newInput = (float) (input - 0.2)/4 + 0.8f;
            return newInput;
        }
    }
}

Evaluator 估值器

我们通过加速器得到的是动画进度的比例,并不是实际的数值,Evaluator就是转换成实际值的地方。(ofInt(0,600)中加速器拿到的是动画的比例值,估值器根据比例转换成实际的值。AnimatorUpdateListener监听器中拿到的实际数值,就是通过Evaluator转换后的值。)

  • 看下面系统Evaluator代码:
//设置取值区间时setIntValues(0,600)/setFloatValues(0,600),这是系统方法,会自动为我们设置相应的类型的Evaluator。

public class IntEvaluator implements TypeEvaluator<Integer> {
    public Integer evaluate(float fraction, Integer startValue, Integer endValue) {
    
        //fraction是在插值器中处理后的进度,在这里以此转换成实际值。
        int startInt = startValue;
        return (int)(startInt + fraction * (endValue - startInt));
    }
}
  • 开下面自定义Evaluator代码:
public class PointEvaluator implements TypeEvaluator<Point> {
    @Override
    public Point evaluate(float fraction, Point startValue, Point endValue) {
        int radius = (int) (startValue.getRadius() + fraction*(endValue.getRadius() - startValue.getRadius()));
        return new Point(radius);
    }
}
public void doAnimation(){
        //ValueAnimator animatior = ValueAnimator.ofObject(new PointEvaluator(), new Point(20), new Point(80));
        //上下作用一样,不可混用。
        ValueAnimator animatior = new ValueAnimator();
        animatior.setObjectValues(new Point(20), new Point(80));
        animatior.setEvaluator(new PointEvaluator());
        
        animatior.setDuration(2000);
        animatior.setInterpolator(new BounceInterpolator());
        animatior.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                mPoint = (Point) animation.getAnimatedValue();
                invalidate();
            }
        });
        animatior.start();
    }

关于取值范围

  • void setObjectValues(Object... values);
  • void setIntValues(int... values);
  • void setFloatValues(float... values);
animatior.setObjectValues(new Point(20), new Point(150),new Point(80), new Point(200), new Point(20), new Point(150));
animatior.setDuration(2000);

public class PointEvaluator implements TypeEvaluator<Point> {
    @Override
    public Point evaluate(float fraction, Point startValue, Point endValue) {
        int radius = (int) (startValue.getRadius() + fraction*(endValue.getRadius() - startValue.getRadius()));
        return new Point(radius);
    }
}

看上面的代码可以发现,setObjetValue可以有n个值。但是Evaluator估值器中只有startValue和endValue。

               26.698   evaluate: startPoint=20-endPoint=150-fraction=0.0
               ......
               26.979   evaluate: startPoint=20-endPoint=150-fraction=0.994
               
               26.995   evaluate: startPoint=150-endPoint=80-fraction=0.049999993
               27.245   evaluate: startPoint=150-endPoint=80-fraction=0.92499995
               27.262   evaluate: startPoint=150-endPoint=80-fraction=0.9844998
               
               27.278   evaluate: startPoint=80-endPoint=200-fraction=0.043999884
               ......
               27.546   evaluate: startPoint=80-endPoint=200-fraction=0.975
               
               27.562   evaluate: startPoint=200-endPoint=20-fraction=0.03449991
               ......
               27.828   evaluate: startPoint=200-endPoint=20-fraction=0.9689996
               
               27.847   evaluate: startPoint=20-endPoint=150-fraction=0.02499974
               ......
               28.112   evaluate: startPoint=20-endPoint=150-fraction=0.9595001
               
               28.129   evaluate: startPoint=150-endPoint=80-fraction=0.018999936
               ......
               28.179   evaluate: startPoint=150-endPoint=80-fraction=0.99399978

ke以发现,设置取值范围的时候,有几个参数,就把setDuration(2000)的时间分为多少分,1&2/2&3 / 3&4...作为start/endValue。至于每份取得的时间比例,就看设置的插值器了。


tips

  • ArgbEvalutor 颜色估值器。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容