Android MotionLayout аварийно завершает работу при переходе с альфа-2 на альфа-3

Я играю с 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, вызывающие этот сбой:

Как вы можете видеть в альфа-3 this.mDevModeDraw.draw(canvas, this.mFrameArrayList, this.mScene.getDuration(), this.mDebugPath); будет вызываться и вызывать сбой при попытке получить продолжительность сцены движения this.mScene.getDuration(), но в альфа-2 он не будет вызван из-за другого условия if.

Как мне решить это?

4 ответа

Вы не опубликовали свой XML, хотя можете проверить, есть ли у вас app:layoutDescription атрибут на макете набора.

Для меня это случилось, когда я подкласс 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)
}

...

Исправлена ​​та же проблема путем обновления библиотеки из

implementation 'androidx.constraintlayout:constraintlayout:2.0.1'

к

implementation 'androidx.constraintlayout:constraintlayout:2.0.4'

Основной причиной этого конкретного сбоя было изменение в AAPT, обнаружившее ошибку в новых атрибутах ConstraintLayout . Это было исправлено в одной из бета-версий, поэтому, если вы начинаете работать с MotionLayout сейчас, вам следует просто использовать последнюю стабильную версию. Если по какой-то причине вы застряли на старой альфа-версии, вы можете поместить их в свой 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.

Другие вопросы по тегам