#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.