Android Animation实现元素在屏幕上按照指定轨迹运动,以及出现NullPointerException的解决方案

原创 2015年11月19日 14:19:35

因项目需要,在Android中实现了一个动画,当在Activity中点击特定按钮时,会在屏幕上添加一个ImageView,并按照指定的起点、终点,沿着特定的轨迹运动(例如直线)。

实现方法

实现思路是在Activity的DecorView中添加一个FrameLayout,然后在FrameLayout中添加ImageView,可通过Margin参数指定ImageView的起始位置。然后设置ImageView的动画,使其能运动到终点。当动画结束后,移除FrameLayout。

核心代码如下。项目GitHub链接 https://github.com/jzj1993/AnimationCrash

  1. public static boolean startAnim(Activity activity, int fromX, int toX, int fromY, int toY) {
  2. try {
  3. final ImageView img = new ImageView(activity);
  4. img.setImageResource(R.mipmap.ic_launcher);
  5. final FrameLayout tempLayout = new FrameLayout(activity);
  6. final FrameLayout.LayoutParams lp = new FrameLayout.LayoutParams(
  7. FrameLayout.LayoutParams.WRAP_CONTENT, FrameLayout.LayoutParams.WRAP_CONTENT);
  8. lp.setMargins(fromX, fromY, 0, 0);
  9. tempLayout.addView(img, lp);
  10. final ViewGroup container = (ViewGroup) activity.getWindow().getDecorView();
  11. container.addView(tempLayout, new ViewGroup.LayoutParams(
  12. ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT));
  13. final Animation anim = new TranslateAnimation(0, toX - fromX, 0, toY - fromY);
  14. anim.setDuration(500);
  15. anim.setAnimationListener(new Animation.AnimationListener() {
  16. @Override
  17. public void onAnimationStart(Animation animation) {
  18. }
  19. @Override
  20. public void onAnimationEnd(Animation animation) {
  21. container.removeView(tempLayout);
  22. }
  23. @Override
  24. public void onAnimationRepeat(Animation animation) {
  25. }
  26. });
  27. img.startAnimation(anim);
  28. } catch (Exception e) {
  29. e.printStackTrace();
  30. }
  31. return true;
  32. }

错误描述

在大部分安卓手机上运行正常,但是在部分Android 4.0.x系统中(例如Nexus S Android 4.0.3模拟器),如果连续同时执行多个动画,可能会出现NullPointerException,如下:

  1. com.jzj1993.anim E/AndroidRuntime FATAL EXCEPTION: main
  2. java.lang.NullPointerException
  3. at android.view.ViewGroup.dispatchDraw(ViewGroup.java:2488)
  4. at android.view.View.draw(View.java:10981)
  5. at android.widget.FrameLayout.draw(FrameLayout.java:450)
  6. at com.android.internal.policy.impl.PhoneWindow$DecorView.draw(PhoneWindow.java:2126)
  7.             at android.view.ViewRootImpl.draw(ViewRootImpl.java:2026)
  8.             at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1634)
  9.             at android.view.ViewRootImpl.handleMessage(ViewRootImpl.java:2442)
  10.             at android.os.Handler.dispatchMessage(Handler.java:99)
  11.             at android.os.Looper.loop(Looper.java:137)
  12.             at android.app.ActivityThread.main(ActivityThread.java:4424)
  13.             at java.lang.reflect.Method.invokeNative(Native Method)
  14.             at java.lang.reflect.Method.invoke(Method.java:511)
  15.             at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
  16. at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
  17. at dalvik.system.NativeStart.main(Native Method)

解决方法

将以下代码

  1. @Override
  2. public void onAnimationEnd(Animation animation) {
  3. container.removeView(tempLayout);
  4. }

改为:

  1. @Override
  2. public void onAnimationEnd(Animation animation) {
  3. container.post(new Runnable() {
  4. @Override
  5. public void run() {
  6. container.removeView(tempLayout);
  7. }
  8. });
  9. }

原因分析

以下是官方给出的ViewGroup.removeView方法的JavaDoc注释:

  1. public void removeView (View view)
  2. Added in API level 1
  3. Note: do not invoke this method from draw(android.graphics.Canvas), onDraw(android.graphics.Canvas), dispatchDraw(android.graphics.Canvas) or any related method.

猜测在同时执行多个动画时,前一个动画的onAnimationEnd被回调时,后一个动画正在执行,因此DecorView的onDraw方法在执行,可能会导致DecorView.onDraw执行时,onAnimationEnd回调中的DecorView.removeView方法被调用,从而导致错误。没有具体深入的去分析源码,但是经过验证问题确实解决了。

版权声明:本文为博主原创文章,转载请注明原文链接。

Android 动画--使用Path来规划动画的轨迹

使用Path来规划动画的轨迹 public void testPathAnimator(){ final FrameLayout l = (FrameLayout) findVi...
  • heqiangflytosky
  • heqiangflytosky
  • 2016年07月11日 10:35
  • 5402

Android利用vectordrawable实现轨迹动画

Google终于在Android5.0中引入了矢量图,矢量图的特点很多,比如不管怎么放大都不会变形,大小也比png小很多,而且利用矢量图我们可以很简单的实现一些看似很复杂的动画效果 虽然动画看...
  • u012835548
  • u012835548
  • 2016年12月14日 15:20
  • 1539

Android动画机制与使用技巧(五)——Android 5.X SVG 矢量动画机制

Google在Android 5.X 中增加了对SVG 矢量图形的支持,这对于创建新的高效率动画具有非常重大的意义。...
  • tw19911005
  • tw19911005
  • 2016年06月05日 17:58
  • 7350

百度地图——显示小车轨迹动画回放

百度地图,Android显示车辆轨迹动画 初次设计想就用百度自带的显示覆盖物的方式,计算两个坐标点的距离,添加短距离的坐标点,然后在密密麻麻的坐标点之间显示,隐藏覆盖物,形成移动的视觉效果。 后来发现...
  • baidu_16668271
  • baidu_16668271
  • 2016年09月23日 17:28
  • 3113

百度地图API四:实现轨迹动态回放功能

本文记录 百度地图实现实现 轨迹回放的功能。   所谓轨迹回放功能。 是指选择某个时间段,然后在地图上再现这个时间段内轨迹的出现情况。 什么时候出现了第几个轨迹点。隔几秒出现下一个轨迹 点。...
  • liusaint1992
  • liusaint1992
  • 2015年11月28日 15:43
  • 17846

Path+ObjectAnimator实现炫酷的轨迹动画

Android5.0以后出了很多好玩的东西。如轨迹动画,矢量图动画等。。。今天我玩的这个东西是把属性动画ObjectAnimator和Path进行结合实现了几个小效果,只要懂得举一反三其他的炫效果都能...
  • Jiang_Rong_Tao
  • Jiang_Rong_Tao
  • 2017年02月28日 20:10
  • 1544

Android动画(实现抛物线运动)

Android动画实现包括视图动画以及属性动画。其中比较新颖的自然是5.X系统下的矢量图动画,这个小编也仅仅只是做了初步的了解,毕竟连矢量图的生成还没有掌握好,并且漂亮的矢量图也必须借助工具来实现。本...
  • u013523377
  • u013523377
  • 2016年04月05日 11:43
  • 3461

安卓沿Path运动轨迹

当然是先看效果拉~中间那个圆圈,顺着圆圈的轨迹运动。 实现一个跟随Path路径运动的效果 思路 1.绘制一个路径 2.得到路径的每个点的坐标 3.用属性动画沿着路径点的坐标运动 ...
  • u010335298
  • u010335298
  • 2016年03月23日 11:20
  • 1294

Android动画(实现抛物线运动)

Android动画实现包括视图动画以及属性动画。其中比较新颖的自然是5.X系统下的矢量图动画,这个小编也仅仅只是做了初步的了解,毕竟连矢量图的生成还没有掌握好,并且漂亮的矢量图也必须借助工具来实现。本...
  • u013523377
  • u013523377
  • 2016年04月05日 11:43
  • 3461

android自定义控件---小球圆周运动

android 小球圆周 运动
  • liubo080852
  • liubo080852
  • 2015年01月17日 16:51
  • 2724
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Android Animation实现元素在屏幕上按照指定轨迹运动,以及出现NullPointerException的解决方案
举报原因:
原因补充:

(最多只允许输入30个字)