【问题标题】:Android MotionLayout crashes when migrating from alpha-2 to alpha-3从 alpha-2 迁移到 alpha-3 时,Android MotionLayout 崩溃
【发布时间】:2019-06-10 09:10:48
【问题描述】:

我在玩 MotionLayout,遇到了一个奇怪的问题

当我将使用的库版本从 com.android.support.constraint:constraint-layout:2.0.0-alpha2 更改为 com.android.support.constraint:constraint-layout:2.0.0-alpha3

我的应用崩溃了:

E/AndroidRuntime: FATAL EXCEPTION: main
Process: ru.sergik.collapsibleexoplayer, PID: 1376
java.lang.NullPointerException: Attempt to invoke virtual method 'int androidx.constraintlayout.motion.widget.MotionScene.getDuration()' on a null object reference
    at androidx.constraintlayout.motion.widget.MotionLayout.dispatchDraw(MotionLayout.java:1634)
    at android.view.View.updateDisplayListIfDirty(View.java:19306)
    at android.view.View.draw(View.java:20093)
    at android.view.ViewGroup.drawChild(ViewGroup.java:4421)
    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:4207)
    at androidx.constraintlayout.widget.ConstraintLayout.dispatchDraw(ConstraintLayout.java:2072)
    at android.view.View.updateDisplayListIfDirty(View.java:19306)
    at android.view.View.draw(View.java:20093)
    at android.view.ViewGroup.drawChild(ViewGroup.java:4421)
    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:4207)
    at android.view.View.updateDisplayListIfDirty(View.java:19306)
    at android.view.View.draw(View.java:20093)
    at android.view.ViewGroup.drawChild(ViewGroup.java:4421)
    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:4207)
    at android.view.View.updateDisplayListIfDirty(View.java:19306)
    at android.view.View.draw(View.java:20093)
    at android.view.ViewGroup.drawChild(ViewGroup.java:4421)
    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:4207)
    at android.view.View.updateDisplayListIfDirty(View.java:19306)
    at android.view.View.draw(View.java:20093)
    at android.view.ViewGroup.drawChild(ViewGroup.java:4421)
    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:4207)
    at android.view.View.updateDisplayListIfDirty(View.java:19306)
    at android.view.View.draw(View.java:20093)
    at android.view.ViewGroup.drawChild(ViewGroup.java:4421)
    at android.view.ViewGroup.dispatchDraw(ViewGroup.java:4207)
    at android.view.View.draw(View.java:20373)
    at com.android.internal.policy.DecorView.draw(DecorView.java:980)
    at android.view.View.updateDisplayListIfDirty(View.java:19315)
    at android.view.ThreadedRenderer.updateViewTreeDisplayList(ThreadedRenderer.java:686)
    at android.view.ThreadedRenderer.updateRootDisplayList(ThreadedRenderer.java:692)
    at android.view.ThreadedRenderer.draw(ThreadedRenderer.java:800)
    at android.view.ViewRootImpl.draw(ViewRootImpl.java:3496)
    at android.view.ViewRootImpl.performDraw(ViewRootImpl.java:3283)
    at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:2818)
    at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1780)
    at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:7827)
    at android.view.Choreographer$CallbackRecord.run(Choreographer.java:911)
    at android.view.Choreographer.doCallbacks(Choreographer.java:723)
    at android.view.Choreographer.doFrame(Choreographer.java:658)
    at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:897)
    at android.os.Handler.handleCallback(Handler.java:789)
    at android.os.Handler.dispatchMessage(Handler.java:98)
    at android.os.Looper.loop(Looper.java:164)
    at android.app.ActivityThread.main(ActivityThread.java:6944)
    at java.lang.reflect.Method.invoke(Native Method)
    at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:327)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1374)

在进行一些调试时,我发现 MotionLayout 的版本存在一些差异导致此崩溃:

正如您在 alpha-3 中看到的,this.mDevModeDraw.draw(canvas, this.mFrameArrayList, this.mScene.getDuration(), this.mDebugPath); 将在尝试获取运动场景持续时间this.mScene.getDuration() 时被调用并导致崩溃,但在 alpha-2 中,由于 if 条件不同,它不会被调用。

我应该如何解决这个问题?

【问题讨论】:

  • 为了提供帮助,我们需要了解您如何使用 MotionLayout - 尤其是您的 MotionScene。
  • 对我来说,这可能发生在我使用合并标签并用它膨胀我自己的 MotionLayout 子类时。但是在 alpha2 上,这不会崩溃,但也不起作用。

标签: android android-constraintlayout android-motionlayout


【解决方案1】:

虽然您可以检查布局集上是否有 app:layoutDescription 属性,但您尚未发布您的 XML。

【讨论】:

    【解决方案2】:

    对我来说,当我将MotionLayout 子类化并使用<merge... 标签夸大其内容时,就会发生这种情况。忘记了合并标签的参数被忽略了,你必须在代码中手动应用它们。

    所以现在看起来像这样:

    class SomeLayout : MotionLayout {
    
    constructor(context: Context, attrs: AttributeSet?, defStyleAttr: Int) : super(context, attrs, defStyleAttr) {
        inflate(context, R.layout.some_layout, this)
        loadLayoutDescription(R.xml.some_motion_scene)
        setTransition(R.id.start, R.id.end)
    }
    
    ...
    

    【讨论】:

      【解决方案3】:

      通过更新库修复了相同的问题 来自

      implementation 'androidx.constraintlayout:constraintlayout:2.0.1'
      

      implementation 'androidx.constraintlayout:constraintlayout:2.0.4'
      

      【讨论】:

        【解决方案4】:

        这次特定崩溃的根本原因是发现 a bug in ConstraintLayout's new attributes 的 AAPT 发生了变化。这已在其中一个测试版中得到修复,因此如果您现在进入 MotionLayout,您应该只使用最新的稳定版本。如果由于某种原因您卡在旧的 alpha 版本中,您可以将它们放入您的 values/attrs.xml 文件中以解决崩溃问题:

            <attr name="duration"/>
            <attr name="flow_horizontalSeparator"/>
            <attr name="flow_verticalSeparator"/>
            <attr name="motionPathRotate"/>
            <attr name="motionProgress"/>
            <attr name="waveDecay"/>
        

        强烈建议改用最新的稳定版本 2.0.4。

        【讨论】:

          猜你喜欢
          • 2023-04-05
          • 2020-10-12
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 2017-05-03
          • 1970-01-01
          • 2019-09-06
          • 2011-07-18
          相关资源
          最近更新 更多