Привязка данных не работает в модуле динамических объектов, вызывая исключение нулевого указателя при обращении к представлениям привязки

#android #android-databinding #dynamic-feature-module

#Android #android-привязка данных #dynamic-feature-module

Вопрос:

У меня есть вызываемый модуль динамических функций favorite . Этот модуль поставляется во время установки. В этом модуле я использую привязку данных. Однако, когда я хочу получить доступ к идентификатору представления, он говорит, что представление равно нулю.

Взгляните на XML-макет, присутствующий в моем favorite модуле, который я использую.

 <?xml version="1.0" encoding="utf-8"?>
<layout xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:android="http://schemas.android.com/apk/res/android">

    <data />

    <androidx.constraintlayout.widget.ConstraintLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="?android:colorBackground">

        <com.google.android.material.tabs.TabLayout
            android:id="@ id/frag_favorite_tab_layout"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:elevation="@dimen/keyline_0"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toTopOf="parent" />

        <androidx.viewpager2.widget.ViewPager2
            android:id="@ id/frag_favorite_view_pager"
            android:layout_width="0dp"
            android:layout_height="0dp"
            app:layout_constraintBottom_toBottomOf="parent"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@id/frag_favorite_tab_layout" />
    </androidx.constraintlayout.widget.ConstraintLayout>
</layout>
 

Как вы можете видеть, там есть два представления с соответствующими идентификаторами. Затем я хочу сослаться на эти представления в моем favorite фрагменте модуля. Я сделал это так.

 class FavoriteFragment : BaseFragment() {
    private lateinit var binding: FragFavoriteBinding
    private val viewmodel: FavoriteViewModel by viewModel()
    override var bottomNavigationViewVisibility = View.VISIBLE

    override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View {
        binding = FragFavoriteBinding.inflate(inflater, container, false)
        binding.lifecycleOwner = this
        loadKoinModules(viewModelModule)
        return binding.root
    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        setupViewPager()
        observeNavigationEvent()
    }

    private fun setupViewPager() {
        val viewPagerAdapter =
            FavoriteViewPagerAdapter(
                childFragmentManager,
                viewLifecycleOwner.lifecycle
            )
        // THIS PART OVER HERE v
        binding.fragFavoriteViewPager.adapter = viewPagerAdapter
        TabLayoutMediator(binding.fragFavoriteTabLayout, binding.fragFavoriteViewPager) { tab, position ->
            when (position) {
                0 -> tab.text = getString(ResR.string.movie_toolbar_text)
                1 -> tab.text = getString(ResR.string.show_toolbar_text)
            }
        }.attach()
        binding.fragFavoriteViewPager.registerOnPageChangeCallback(object: ViewPager2.OnPageChangeCallback() {
            override fun onPageScrollStateChanged(state: Int) {
                super.onPageScrollStateChanged(state)
                if (state == ViewPager.SCROLL_STATE_DRAGGING || state == ViewPager.SCROLL_STATE_SETTLING)
                    EspressoIdlingResource.increment()
                else
                    EspressoIdlingResource.decrement()
            }
        })
    }

    private fun observeNavigationEvent() {
        viewmodel.navigateToDetails.observe(viewLifecycleOwner, Observer {
            if (it.first != -1) {
                val action =
                    if (it.second == ListFragment.TYPE_MOVIE) FavoriteFragmentDirections.actionFavoriteFragmentToMovieDetailsFragment2(
                        it.first
                    )
                    else FavoriteFragmentDirections.actionFavoriteFragmentToShowDetailsFragment2(it.first)
                findNavController().navigate(action)
                viewmodel.finishNavigatingToDetails()
            }
        })
    }
}
 

Я ссылаюсь на ViewPager, используя binding.fragFavoriteViewPager , как видно выше. Во время сборки ошибки нет. Однако во время выполнения, когда я пытаюсь перейти к этому FavoriteFragment , он возвращает ошибку исключения нулевого указателя. Ошибка сообщает мне, что binding.fragFavoriteViewPager это нулевая ссылка на объект. Полную ошибку можно увидеть ниже.

 E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.mobile.moviecatalogue, PID: 13131
    java.lang.NullPointerException: binding.fragFavoriteViewPager must not be null
        at com.mobile.favorite.ui.FavoriteFragment.setupViewPager(FavoriteFragment.kt:50)
        at com.mobile.favorite.ui.FavoriteFragment.onViewCreated(FavoriteFragment.kt:40)
        at androidx.fragment.app.FragmentStateManager.createView(FragmentStateManager.java:332)
        at androidx.fragment.app.FragmentManager.moveToState(FragmentManager.java:1199)
        at androidx.fragment.app.FragmentManager.addAddedFragments(FragmentManager.java:2236)
        at androidx.fragment.app.FragmentManager.executeOpsTogether(FragmentManager.java:2009)
        at androidx.fragment.app.FragmentManager.removeRedundantOperationsAndExecute(FragmentManager.java:1965)
        at androidx.fragment.app.FragmentManager.execPendingActions(FragmentManager.java:1861)
        at androidx.fragment.app.FragmentManager$4.run(FragmentManager.java:413)
        at android.os.Handler.handleCallback(Handler.java:790)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:164)
        at android.app.ActivityThread.main(ActivityThread.java:6494)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:438)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:807)E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.mobile.moviecatalogue, PID: 13131
    java.lang.NullPointerException: binding.fragFavoriteViewPager must not be null
        at com.mobile.favorite.ui.FavoriteFragment.setupViewPager(FavoriteFragment.kt:50)
        at com.mobile.favorite.ui.FavoriteFragment.onViewCreated(FavoriteFragment.kt:40)
        at androidx.fragment.app.FragmentStateManager.createView(FragmentStateManager.java:332)
        at androidx.fragment.app.FragmentManager.moveToState(FragmentManager.java:1199)
        at androidx.fragment.app.FragmentManager.addAddedFragments(FragmentManager.java:2236)
        at androidx.fragment.app.FragmentManager.executeOpsTogether(FragmentManager.java:2009)
        at androidx.fragment.app.FragmentManager.removeRedundantOperationsAndExecute(FragmentManager.java:1965)
        at androidx.fragment.app.FragmentManager.execPendingActions(FragmentManager.java:1861)
        at androidx.fragment.app.FragmentManager$4.run(FragmentManager.java:413)
        at android.os.Handler.handleCallback(Handler.java:790)
        at android.os.Handler.dispatchMessage(Handler.java:99)
        at android.os.Looper.loop(Looper.java:164)
        at android.app.ActivityThread.main(ActivityThread.java:6494)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:438)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:807)
 

I have set up my Gradle file for that favorite module like so.

 plugins {
    id 'com.android.dynamic-feature'
    id 'kotlin-android'
    id 'kotlin-kapt'
}

apply from: '../shared_dependencies.gradle'

android {
    compileSdkVersion 30
    buildToolsVersion "30.0.2"

    defaultConfig {
        applicationId "com.mobile.favorite"
        minSdkVersion 27
        targetSdkVersion 30
        versionCode 1
        versionName "1.0"

        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
    }

    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }
    buildFeatures {
        dataBinding true
    }
    compileOptions {
        sourceCompatibility JavaVersion.VERSION_1_8
        targetCompatibility JavaVersion.VERSION_1_8
    }
    kotlinOptions {
        jvmTarget = '1.8'
    }
}

dependencies {
    implementation project(":core")
    implementation project(":app")
    implementation project(":res")
}
 

Для перехода к этому фрагменту динамического объекта в моем другом вызываемом модуле app у меня есть этот график навигации.

 <?xml version="1.0" encoding="utf-8"?>
<navigation xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@ id/main_navigation"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    app:startDestination="@id/movieFragment">

    ...
    <fragment
        app:moduleName="favorite"
        android:id="@ id/favoriteFragment"
        android:name="com.mobile.favorite.ui.FavoriteFragment"
        android:label="@string/favorite_toolbar_text"
        tools:layout="@layout/frag_favorite">
        <action
            android:id="@ id/action_favoriteFragment_to_movieDetailsFragment2"
            app:destination="@id/movieDetailsFragment"
            app:enterAnim="@anim/nav_default_enter_anim"
            app:exitAnim="@anim/nav_default_exit_anim"
            app:popEnterAnim="@anim/nav_default_pop_enter_anim"
            app:popExitAnim="@anim/nav_default_pop_exit_anim" />
        <action
            android:id="@ id/action_favoriteFragment_to_showDetailsFragment2"
            app:destination="@id/showDetailsFragment"
            app:enterAnim="@anim/nav_default_enter_anim"
            app:exitAnim="@anim/nav_default_exit_anim"
            app:popEnterAnim="@anim/nav_default_pop_enter_anim"
            app:popExitAnim="@anim/nav_default_pop_exit_anim" />
    </fragment>
    ...
    <action
        android:id="@ id/action_global_favoriteFragment"
        app:destination="@id/favoriteFragment"
        app:enterAnim="@anim/nav_default_enter_anim"
        app:exitAnim="@anim/nav_default_exit_anim"
        app:popEnterAnim="@anim/nav_default_pop_enter_anim"
        app:popExitAnim="@anim/nav_default_pop_exit_anim" />
</navigation>
 

Я внес изменения в график навигации на основе документа Android, касающегося динамической навигации. Затем я удалил все коды, связанные с вызовом binding.AnyView и использовал Layout Inspector . Оказывается, что у всех представлений больше не было идентификатора. Смотрите эту картинку ниже.

введите описание изображения здесь

Почему идентификатор исчез в инспекторе компоновки? Что я могу сделать, чтобы иметь возможность ссылаться на представления, используя привязку данных в файле фрагмента?

Редактировать

Используя binding.root.findViewById , я могу получить доступ к представлениям. Это странно.

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

1. Спасибо, что предоставили решение в редактировании, у меня та же проблема, поэтому я буду придерживаться findViewById…

2. @MerthanE Я все еще надеюсь, что кто-то может дать ответ, не используя findViewById , хотя, потому что использование binding должно позволить нам быстрее извлекать представления .

3. даже findViewById не работает в моем коде