共享元素动画在Android10上异常的解决方案(全网唯一仅此一份)

问题描述

最近在项目中遇到一个关于共享元素动画失效的问题,在Activity跳转时使用ActivityOptionsCompat.makeSceneTransitionAnimation做转场动画,
对于Activity共享元素动画的使用不懂的童鞋自行谷歌,网上资料太多,这里就不做描述了。

一般情况下使用ActivityOptionsCompat.makeSceneTransitionAnimation做转场动画时,当Activity返回时也会伴随返回的动画。但是就是这样的一个简单的API,
居然在Android 10上测试出现了异常:

如果使用了ActivityOptionsCompat.makeSceneTransitionAnimation跳转的Activity没有再次跳转到别的Activity的话,返回时系统是带上响应的共享元素动画的,
但是如果跳转的Activity再次跳转了其他的Activity,然后再返回的话共享元素的返回动画就失效了(包括按了Home键退回桌面,然后再次进入也会失效)。

笔者的测试手机是华为nova 5i,系统是Android 10

问题追踪

1、 大神方案

当笔者看到这个bug的时候,首先想到的可能是我打开的方式不对,于是谷歌百度了一波,结果确实无功而返。

于是笔者第一想到的就是是不是自己的使用方式不对,于是看看大神是怎么使用的,然后参考了一下Android 大神郭霖开源的一个开源项目giffun
发现giffun也存在同样的异常…

为此我专门给giffun提了一个issue:

https://github.com/guolindev/giffun/issues/67

绝望了。。。。。。。

2、 竞品对比

既然查找不到相关的资料,那么就看看竞品是否也有这样的问题的,如果竞品也有这样的问题话就可以拿着竞品去忽悠产品了,心里美滋滋。

于是笔者对比了一下竞品的产品是否也存在这么的一个bug,发现竞品没有一样的bug,它们的共享元素动画一切正常。
后面通过研究了它们的APK才发现它们的共享元素动画是使用Fragment做的,所以尽量使用Fragment的一个好处又体现出来了。

但是笔者今天要说的将Activity的转场动画改为Fragment这样就完事了,今天将带大家一步一步分析共享元素动画是如何失效的,如何修复这样的一个bug。

3、 源码先行

既然网上没有这样的资料,那就自己动手,丰衣足食了。首先我们初步看看Activity的源码,发现如果是返回带执行共享元素动画的话执行的方法是finishAfterTransition
finishAfterTransition里面调用了ActivityTransitionStatestartExitBackTransition方法。

然而看源码并没有看出什么破绽,毕竟大多数手机是正常的,只有在Android 10上才出现了异常。既然这条路走不通,那我们就换一个途径吧。

4、 Debug大法

我们使用Android Studio的断点调试功能,在ActivityTransitionStatestartExitBackTransition方法里面打断点调试,
对比发现最终返回时没有执行共享元素动画的原因是startExitBackTransition方法内的pendingExitNames变量为空就直接返回了false,也就是不执行共享元素动画。

5、 万变不离其宗,再次回到源码

那么为什么pendingExitNames变量会变成了空呢?我们再次回到源码分析,pendingExitNames变量是通过
ActivityTransitionStatestartExitBackTransitiongetPendingExitNames方法获取的,方法如下:

    if (mPendingExitNames == null && mEnterTransitionCoordinator != null) {
            mPendingExitNames = mEnterTransitionCoordinator.getPendingExitSharedElementNames();
        }
        return mPendingExitNames;
    }

很简单的代码,如果mEnterTransitionCoordinator是空的话,那么getPendingExitNames方法必定会返回空。

那么我们继续追踪一下mEnterTransitionCoordinator变量是如何赋值的,跟踪发现是在ActivityTransitionStateenterReady(Activity activity)方法
中对mEnterTransitionCoordinator做了赋值操作,然后在ActivityTransitionStateonStop方法中被置空了,而ActivityTransitionStateonStop方法
又被ActivityonStop方法调用了,至此大概就能解析的解析得通为什么经过跳转后Activity的返回共享元素失效了,原来是被Activity的onStop生命周期给影响了。

ActivityTransitionState的enterReady代码:

public void enterReady(Activity activity) {
        if (mEnterActivityOptions == null || mIsEnterTriggered) {
            return;
        }
        mIsEnterTriggered = true;
        mHasExited = false;
        ArrayList sharedElementNames = mEnterActivityOptions.getSharedElementNames();
        ResultReceiver resultReceiver = mEnterActivityOptions.getResultReceiver();
        if (mEnterActivityOptions.isReturning()) {
            restoreExitedViews();
            activity.getWindow().getDecorView().setVisibility(View.VISIBLE);
        }

        // 在这里对mEnterTransitionCoordinator赋值了
        // 在这里对mEnterTransitionCoordinator赋值了
        // 在这里对mEnterTransitionCoordinator赋值了

        mEnterTransitionCoordinator = new EnterTransitionCoordinator(activity,
                resultReceiver, sharedElementNames, mEnterActivityOptions.isReturning(),
                mEnterActivityOptions.isCrossTask());
        if (mEnterActivityOptions.isCrossTask()) {
            mExitingFrom = new ArrayList<>(mEnterActivityOptions.getSharedElementNames());
            mExitingTo = new ArrayList<>(mEnterActivityOptions.getSharedElementNames());
        }

        if (!mIsEnterPostponed) {
            startEnter();
        }
    }

ActivityTransitionState的onStop代码:


  public void onStop() {
        restoreExitedViews();
        if (mEnterTransitionCoordinator != null) {
            mEnterTransitionCoordinator.stop();
            mEnterTransitionCoordinator = null;
        }
        if (mReturnExitCoordinator != null) {
            mReturnExitCoordinator.stop();
            mReturnExitCoordinator = null;
        }
    }

在这里给同学们留一个问题,既然是被Activity的onStop方法影响了,那么为什么有些系统可以,但是在Android 10系统上就不行了呢?这个问题留给大家去解答,因为最近项目非常忙,
笔者也没时间去深究这个问题了。

如何解决

既然找到了问题那么就好解决了,我们在Activity的onResume方法中调用一下ActivityTransitionStateenterReady方法,再次给mEnterTransitionCoordinator赋值不久完事了吗?

但是ActivityTransitionState类是系统的私有类,开发者是不能直接调用的,这时候我们就想到了Java的反射大法,是的笔者就是通过反射调用的。

在编写反射调用ActivityTransitionStateenterReady方法时候AS提示targetSdkVersion是28以上的会得到反射异常。这是因为Android 9以上对反射做了限制,这时候我们只需要将targetSdkVersion设置成28以下的即可。

主要代码:


 /**
     * Android10 Activity的onStop方法可能会导致共享元素动画失效,通过反射注入恢复共享元素动画
     * @param activity
     */
    public static void updateResume(Activity activity){
        try {
            Field activityTransitionStateField = Activity.class.getDeclaredField("mActivityTransitionState");
            activityTransitionStateField.setAccessible(true);
            Object mActivityTransitionState = activityTransitionStateField.get(activity);
            Class clazz = Class.forName("android.app.ActivityTransitionState");
            Method enterReady = clazz.getDeclaredMethod("enterReady",Activity.class);
            enterReady.setAccessible(true);
            enterReady.invoke(mActivityTransitionState,activity);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

然后在Activity的onResume方法中调用一下反射方法即可。实测返回是共享元素动画正常,但是会导致什么未知bug目前还不可知,欢迎大家深究讨论。。。

关注我,一起学习,不止于技术!!!
共享元素动画在Android10上异常的解决方案(全网唯一仅此一份)_第1张图片

你可能感兴趣的:(共享元素动画在Android10上异常的解决方案(全网唯一仅此一份))