NineOldAndroids动画兼容库的使用-ViewHelper

微生宝
2023-12-01

本文转载自:http://www.bkjia.com/Androidjc/936210.html

功能介绍: NineOldAndroids是github上的一个开源项目,其作用是为了在低版本android上(API11以下)使用属性动画。它的原理其实也很简单,主要就是判断当前sdk版本,如果大于API11,那么就调用官方的API,否则自己实现动画效果。另外,在API使用方面,它与官方的属性动画基本一致。比如ObjectAnimator、ValueAnimator等等。

一.基本使用

跟官方类似,NineOldAndroids主要是通过ObjectAnimator、ValueAnimator、AnimatorSet完成属性动画。

示例代码:
1.ObjectAnimator:让按钮水平向右移动100单位

Button btn = ...
ObjectAnimator.ofFloat(btn,"translationX",100).setDuration(2000).start();

2.ValueAnimator: 让按钮竖直向下移动100单位

Button btn = ...
ValueAnimator animator = ValueAnimator.ofFloat(0,100).setDuration(2000);
animator.start();
animator.addUpdateListener(new AnimatorUpdateListener()
{
    @Override
    public void onAnimationUpdate(ValueAnimator animation)
    {
        float curVal = (float) animation.getAnimatedValue();
        ViewHelper.setTranslationY(btn,curVal);//注意不要使用btn.setTranslationY
    }
});

大家应该注意到这里我们使用了一个新的类,叫ViewHelper,这个类是为了兼容以前的api,因为像setAlpha,setTranslationX等方法在低版本中是没有的,所以NineOldAndroids提供了ViewHelper类,使得我们不必关心API版本。

3.AnimatorSet: 水平移动100,竖直移动100,并且透明度发生改变。

Button btn =...
AnimatorSet set = new AnimatorSet();
set.playTogether(
        ObjectAnimator.ofFloat(btn,"translationX",100),
        ObjectAnimator.ofFloat(btn,"translationY",100),
        ObjectAnimator.ofInt(btn,"alpha",1,0)
                );
set.setDuration(2000);
set.start();

当然了,这个效果也可以通过PropertyValuesHolder来实现:

Button btn = ...
ObjectAnimator.ofPropertyValuesHolder(btn,
                PropertyValuesHolder.ofFloat("translationX",100),
                PropertyValuesHolder.ofFloat("translationY",100),
                PropertyValuesHolder.ofInt("alpha",1,0)
                ).start();


4.ViewPropertyAnimator: 我们知道,官方提供了ViewPropertyAnimator类简化了View动画,只要我们调用View提供的animate方法即可返回ViewPropertyAnimator实例,然后我们可以调用其x、translationX、alpha、rotation等方法很方便的实现动画。 NineOldAndroids库也提供了ViewPropertyAnimator类,但是使用方式上有点不一样。它是通过ViewPropertyAnimator提供静态工厂方法animate,并且在参数中必须绑定一个view,然后返回ViewPropertyAnimator实例。 示例代码:

Button btn = ...
ViewPropertyAnimator.animate(btn).translationX(100).translationY(100);


注:以上示例中使用的类应该全部导入nineoldandroids包中的!

二.ViewHelper类原理

ViewHelper提供了一系列静态set/get方法去操作View的各种属性,比如透明度、偏移量、旋转角度等等,这大大方便了我们的使用,并且有个好处就是不用考虑低版本的兼容性问题。那么ViewHelper是如何做到的呢?下面我们简单剖析下原理。

打开ViewHelper类源码,我们随便找一个方法,比如setAlpha方法:

public static void setAlpha(View view, float alpha) {
        if (NEEDS_PROXY) {
            wrap(view).setAlpha(alpha);
        } else {
            Honeycomb.setAlpha(view, alpha);
        }
    }

这个方法内部有个判断,如果需要代理的话,那么就调用代理类的setAlpha方法,否则直接调用HoneyComb的setAlpha方法。这个NEEDS_PROXY是AnimatorProxy中的常量:

 public static final boolean NEEDS_PROXY = Integer.valueOf(Build.VERSION.SDK).intValue() < Build.VERSION_CODES.HONEYCOMB;


判断当前sdk版本是否是API11以下,如果是的话那么NEEDS_PROXY为true。这时候会进入if分支,调用AnimatorProxy类中的wrap方法,将view进行包装,然后调用setAlpha方法设置透明度:

public void setAlpha(float alpha) {
        if (mAlpha != alpha) {
            mAlpha = alpha;
            View view = mView.get();
            if (view != null) {
                view.invalidate();
            }
        }
    }

可以看到这个方法会设置mAlpha变量,并调用invalidate方法刷新视图,刷新过程中,会调用applyTransformation方法更新透明度。

@Override
    protected void applyTransformation(float interpolatedTime, Transformation t) {
        //setAlpha方法会导致view.invalidate刷新,而在刷新过程中,会回调此方法设置透明度
        View view = mView.get();
        if (view != null) {
            t.setAlpha(mAlpha);
            transformMatrix(t.getMatrix(), view);
        }
    }

之所以这样做,是因为低版本没有提供setAlpha方法。如果sdk版本大于11,那么进入else分支,调用HoneyComb.setAlpha:

 static void setAlpha(View view, float alpha) {
            view.setAlpha(alpha);
        }

其他方法都跟这个类似。
通过上面分析相信大家对NineOldAndroids有了更深入的理解了吧!

 类似资料: