在动画正在播放的时候(如果你的动画跟界面的坐标点有关),界面失去隐藏,或者失去了焦点,如果没有处理,就会报这个空指针异常。
复现场景:
自定义了一个父控件,然后里面有一些子控件在执行动画,动画是根据屏幕的宽高移动,这时候,拉下Android的状态栏,或者按home键,甚至退出程序,都会报一个空指针异常,异常如下图。
java.lang.NullPointerException
at android.animation.KeyframeSet.getValue(KeyframeSet.java:183)
at android.animation.PropertyValuesHolder.calculateValue(PropertyValuesHolder.java:660)
at android.animation.ValueAnimator.animateValue(ValueAnimator.java:1174)
at android.animation.ValueAnimator.animationFrame(ValueAnimator.java:1115)
at android.animation.ValueAnimator.doAnimationFrame(ValueAnimator.java:1144)
at android.animation.ValueAnimator$AnimationHandler.doAnimationFrame(ValueAnimator.java:623)
at android.animation.ValueAnimator$AnimationHandler.run(ValueAnimator.java:646)
at android.view.Choreographer$CallbackRecord.run(Choreographer.java:749)
at android.view.Choreographer.doCallbacks(Choreographer.java:562)
at android.view.Choreographer.doFrame(Choreographer.java:531)
at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:735)
at android.os.Handler.handleCallback(Handler.java:730)
at android.os.Handler.dispatchMessage(Handler.java:92)
at android.os.Looper.loop(Looper.java:137)
at android.app.ActivityThread.main(ActivityThread.java:5103)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:525)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
at dalvik.system.NativeStart.main(Native Method)
解决:
从异常上
来看这个跟PropertyValuesHolder有关,但最终原因不确定。我是这么想的,View Animation 底层的动画执行过程依然跟Animator有关,所以在失去焦点后他便找不到坐标点了,所以他的过渡点便成了null。只是这么想,不知道对不对……
所以我在自定义控件的onVisibilityChanged()中,判断了是否是显示状态,如果是显示状态,就让他继续播放动画,否则我就执行动画的cancel(),方法使之先取消。
cancel方法:取消动画
【功能说明】该方法用于取消一个动画的执行。该方法是取得一个正在执行中的动画的主要方法。cancel方法和startNow方法结合可以实现对动画执行过程的控制。需要注意的是,通过cancel方法取消的动画,必须使用reset方法或者setAnimation方法重新设置,才可以再次执行动画。
@Override
protected void onVisibilityChanged(View changedView, int visibility) {
super.onVisibilityChanged(changedView, visibility);
if (visibility != VISIBLE){
View childAt = getChildAt(XXX);
Animation animation = childAt.getAnimation();
animation.cancel();
}else{
View childAt = getChildAt(XXX);
childAt.getAnimation().reset();
childAt.getAnimation().start();
}
}