Прослушиватель перехода Android MotionLayout не вызывается

#android #android-layout #kotlin #android-transitions #android-motionlayout

#Android #android-макет #kotlin #android-переходы #android-motionlayout

Вопрос:

Я пытаюсь создать анимацию слайдов. Для этого я использую MotionLayout с MotionScene, содержащим два ConstraintSet.

Анимация слайдов работает нормально. Однако я получаю сообщение об ошибке в LogCat:

E / MotionLayout: ПРЕДУПРЕЖДЕНИЕ НЕТ приложения: тег layoutDescription

view_slider.xml

 <androidx.constraintlayout.motion.widget.MotionLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@ id/motion_layout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/white"
app:layoutDescription="@xml/scene_slider"
app:showPaths="true">

<View
    android:id="@ id/btn_slide"
    android:layout_width="64dp"
    android:layout_height="64dp"
    android:background="@color/grey26" />
</androidx.constraintlayout.motion.widget.MotionLayout>
  

scene_slider.xml:

 <MotionScene xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:motion="http://schemas.android.com/apk/res-auto">

    <Transition
        motion:constraintSetStart="@ id/start"
        motion:constraintSetEnd="@ id/end"
        motion:duration="1000">

        <OnSwipe
            motion:touchAnchorId="@ id/btn_slide"
            motion:touchAnchorSide="right"
            motion:dragDirection="dragRight"/>

    </Transition>

    <ConstraintSet android:id="@ id/start">
        <Constraint
            android:id="@ id/btn_slide"
            android:layout_width="64dp"
            android:layout_height="64dp"
            android:layout_marginStart="8dp"
            motion:layout_constraintBottom_toBottomOf="parent"
            motion:layout_constraintStart_toStartOf="parent"
            motion:layout_constraintTop_toTopOf="parent" />
    </ConstraintSet>

    <ConstraintSet android:id="@ id/end">
        <Constraint
            android:id="@ id/btn_slide"
            android:layout_width="64dp"
            android:layout_height="64dp"
            android:layout_marginEnd="8dp"
            motion:layout_constraintBottom_toBottomOf="parent"
            motion:layout_constraintEnd_toEndOf="parent"
            motion:layout_constraintTop_toTopOf="parent" />
    </ConstraintSet>

</MotionScene>
  

SliderView.kt:

 class SliderView @JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet? = null,
    defStyleAttr: Int = 0
) : MotionLayout(context, attrs, defStyleAttr) {

    init {
        LayoutInflater.from(context).inflate(R.layout.view_slider, this)
    }

    override fun onFinishInflate() {
        super.onFinishInflate()

        motion_layout.setTransitionListener(object : MotionLayout.TransitionListener {
            override fun onTransitionCompleted(motionLayout: MotionLayout?, currentId: Int) {
                debug(javaClass.simpleName, "transition completed")
            }

            override fun onTransitionChange(motionLayout: MotionLayout?, startId: Int, endId: Int, progress: Float) {
                debug(javaClass.simpleName, "progress = $progress")
            }
        })
    }
}
  

fragment.xml

 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:background="@color/red">

    <com.myapp.utils.views.SliderView
        android:id="@ id/slider"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>

</LinearLayout>
  

Возможно, я что-то упускаю, потому что в коде есть layoutDescription для MotionLayout.

Как исправить эту ошибку и почему теперь вызываются обратные вызовы TransitionListener?

Комментарии:

1. У меня также была проблема, из-за которой TransitionListener не вызывался, когда я менял сцену из кода, анимация воспроизводится / работает? Можем ли мы увидеть код, в котором вы устанавливаете или расширяете представление

2. Да, анимация работает. Я могу сдвинуть btn_slide. Путь также отображается правильно. Я отредактировал свой вопрос, чтобы показать полный код моего пользовательского представления.

3. Попробуйте заменить motion_layout.setTransitionListener() На this.setTransitionListener(), кроме того, у вас, вероятно, есть ссылка из Activity / Fragment, попробуйте установить TransitionListener оттуда

4. Я только что попробовал два ваших предложения, но, к сожалению, в LogCat есть та же ошибка, и прослушиватель не вызывается

5. При вводе MotionLayout непосредственно в мой фрагмент не отображается сообщение об ошибке, однако прослушиватель по-прежнему не вызывается…

Ответ №1:

 var motionLayout:MotionLayout?=null
 init {
       motionLayout =  LayoutInflater.from(context).inflate(R.layout.view_slider, this)
    }
  

Строка выше должна быть приравнена к именованному view..eg описание движения вала.

Эта именованная переменная должна использоваться в onFinish inflate .. т.е.

   override fun onFinishInflate() {
        super.onFinishInflate()

        motionLayout?.setTransitionListener(object : MotionLayout.TransitionListener {
            override fun onTransitionCompleted(motionLayout: MotionLayout?, currentId: Int) {
                debug(javaClass.simpleName, "transition completed")
            }

            override fun onTransitionChange(motionLayout: MotionLayout?, startId: Int, endId: Int, progress: Float) {
                debug(javaClass.simpleName, "progress = $progress")
            }
        })
    }
  

Ответ №2:

MotionLayout не работал должным образом во фрагментах. Попробуйте перейти на бета-версию 2.