Рукоятка кинжала @AndroidEntryPoint не работает с фрагментом нижней таблицы()

#android #gradle #dagger-hilt #bottomsheetdialogfragment

Вопрос:

Я использую ViewModel внутри BottomSheetDialogFragmet (), поэтому я должен пометить свой нижний лист с помощью @AndroidEntryPoint.

 @AndroidEntryPoint
class SearchAddressDialog : BottomSheetDialogFragment() {

    private val viewModel: MyAddressesViewModel by viewModels()
    
    ......
}
 

Но когда я попытался создать свой проект, он выдает мне эту ошибку:

 Execution failed for task ':app:kaptDebugKotlin'.
> A failure occurred while executing 
org.jetbrains.kotlin.gradle.internal.KaptWithoutKotlincTask$KaptExecutionWorkAction
> java.lang.reflect.InvocationTargetException (no error message)

* Try:
Run with --stacktrace option to get the stack trace. Run with --info or --debug option 
to get more log output. Run with --scan to get full insights.
 

И вот мой вывод для запуска с —stacktrace

https://gist.github.com/javlonrahimov/95de968645cace1dfb6e425381f8014b

Если я удалю @AndroidEntryPoint над моей нижней таблицей Fragmen (), это приведет к созданию и установке приложения. Но когда я пытаюсь открыть нижний лист, приложение выходит из строя, так как я ввел свои репозитории в ViewModel:

 java.lang.RuntimeException: Cannot create an instance of class uz.unical.other.ui.my_addresses.view_model.MyAddressesViewModel
    at androidx.lifecycle.ViewModelProvider$NewInstanceFactory.create(ViewModelProvider.java:221)
    at androidx.lifecycle.ViewModelProvider$AndroidViewModelFactory.create(ViewModelProvider.java:278)
    at androidx.lifecycle.SavedStateViewModelFactory.create(SavedStateViewModelFactory.java:112)
    at androidx.lifecycle.ViewModelProvider.get(ViewModelProvider.java:185)
    at androidx.lifecycle.ViewModelProvider.get(ViewModelProvider.java:150)
    at androidx.lifecycle.ViewModelLazy.getValue(ViewModelProvider.kt:54)
    at androidx.lifecycle.ViewModelLazy.getValue(ViewModelProvider.kt:41)
    at uz.unical.other.ui.my_addresses.dialogs.SearchAddressDialog.getViewModel(SearchAddressDialog.kt:26)
    at uz.unical.other.ui.my_addresses.dialogs.SearchAddressDialog.onResume(SearchAddressDialog.kt:86)
    at androidx.fragment.app.Fragment.performResume(Fragment.java:3039)
    at androidx.fragment.app.FragmentStateManager.resume(FragmentStateManager.java:607)
    at androidx.fragment.app.FragmentStateManager.moveToExpectedState(FragmentStateManager.java:306)
    at androidx.fragment.app.FragmentManager.executeOpsTogether(FragmentManager.java:2189)
    at androidx.fragment.app.FragmentManager.removeRedundantOperationsAndExecute(FragmentManager.java:2100)
    at androidx.fragment.app.FragmentManager.execPendingActions(FragmentManager.java:2002)
    at androidx.fragment.app.FragmentManager$5.run(FragmentManager.java:524)
    at android.os.Handler.handleCallback(Handler.java:938)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:246)
    at android.app.ActivityThread.main(ActivityThread.java:8528)
    at java.lang.reflect.Method.invoke(Native Method)
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:602)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1130)
 Caused by: java.lang.InstantiationException: java.lang.Class<uz.unical.other.ui.my_addresses.view_model.MyAddressesViewModel> has no zero argument constructor
    at java.lang.Class.newInstance(Native Method)
    at androidx.lifecycle.ViewModelProvider$NewInstanceFactory.create(ViewModelProvider.java:219)
    at androidx.lifecycle.ViewModelProvider$AndroidViewModelFactory.create(ViewModelProvider.java:278) 
    at androidx.lifecycle.SavedStateViewModelFactory.create(SavedStateViewModelFactory.java:112) 
    at androidx.lifecycle.ViewModelProvider.get(ViewModelProvider.java:185) 
    at androidx.lifecycle.ViewModelProvider.get(ViewModelProvider.java:150) 
    at androidx.lifecycle.ViewModelLazy.getValue(ViewModelProvider.kt:54) 
    at androidx.lifecycle.ViewModelLazy.getValue(ViewModelProvider.kt:41) 
    at uz.unical.other.ui.my_addresses.dialogs.SearchAddressDialog.getViewModel(SearchAddressDialog.kt:26) 
    at uz.unical.other.ui.my_addresses.dialogs.SearchAddressDialog.onResume(SearchAddressDialog.kt:86) 
    at androidx.fragment.app.Fragment.performResume(Fragment.java:3039) 
    at androidx.fragment.app.FragmentStateManager.resume(FragmentStateManager.java:607) 
    at androidx.fragment.app.FragmentStateManager.moveToExpectedState(FragmentStateManager.java:306) 
    at androidx.fragment.app.FragmentManager.executeOpsTogether(FragmentManager.java:2189) 
    at androidx.fragment.app.FragmentManager.removeRedundantOperationsAndExecute(FragmentManager.java:2100) 
    at androidx.fragment.app.FragmentManager.execPendingActions(FragmentManager.java:2002) 
    at androidx.fragment.app.FragmentManager$5.run(FragmentManager.java:524) 
    at android.os.Handler.handleCallback(Handler.java:938) 
    at android.os.Handler.dispatchMessage(Handler.java:99) 
    at android.os.Looper.loop(Looper.java:246) 
    at android.app.ActivityThread.main(ActivityThread.java:8528) 
    at java.lang.reflect.Method.invoke(Native Method) 
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:602) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1130) 
 

@AndroidEntryPoint работает без ошибок на других обычных фрагментах.

Обновить:

Вот моя модель представления:

 @HiltViewModel
class MyAddressesViewModel @Inject constructor(
   private val repository: AddressRepository,
   private val geocoderRepository: GeocoderRepository
) : ViewModel() {}
 

ОБНОВЛЕНИЕ 2:
Прямо сейчас я аннотирую BottomSheetDialogFragment (), и он выдает мне ошибки при построении.

И если я изменю это

 @AndroidEntryPoint
class SearchAddressDialog : BottomSheetDialogFragment(){
 

к этому

 @AndroidEntryPoint
class SearchAddressDialog : Fragment() {
 

он работает совершенно нормально.

Могу ли я на самом деле аннотировать BottomSheetDialogFragment() с помощью @AndroidEntryPoint?

ОБНОВЛЕНИЕ 3:

Вот мой фрагмент нижней таблицы

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

1. Кажется, это правильно, можете ли вы очистить сборку и построить снова?

2. Я попробовал чистую сборку. Но безуспешно.

3. какова ваша версия навигации?

4. const val navigation_version = "2.3.5"

5. вы снабдили модель представления комментариями: @HiltViewModel ?

Ответ №1:

После долгих исследований я не смог решить проблему @AndroidEntryPoint. И я пришел к такому выводу: я просто передаю свою модель представления в свой фрагмент нижней таблицы в его конструкторе:

 class SearchAddressDialog(
    private val viewModel: MyAddressesViewModel
) : BottomSheetDialogFragment() {
    ........ 
}
 

И я открыл свой диалог вот так:

 val dialog = SearchAddressDialog(viewModel)
dialog.show(childFragmentManager, SearchAddressDialog.TAG)
 

Я использовал модель представления текущего фрагмента.

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

1. Почти уверен, что вам придется использовать FragmentFactory , если вы еще этого не сделали, если не попытаетесь повернуть свое устройство (создать изменение конфигурации), я ожидаю, что оно выдаст ошибку, так как FragmentManager не будет знать, как воссоздать фрагмент.

2. Да, я реализовал FragmentFactory после некоторых сбоев :). И теперь он работает идеально.

3. Я нахожу этот ответ наиболее полезным !

Ответ №2:

Вы добавили @Inject аннотацию на
AddressRepository а GeocoderRepository конструкторы классов? возможно, вы забыли аннотировать конструктор зависимого объекта вашего класса ViewModel. из-за этого Hilt/Dagger не может разрешить дерево зависимостей.

Если это так, аннотируйте все зависимые конструкторы классов @Inject , чтобы проблема была решена.

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

1. Я проверил все зависимые объекты и думаю, что все в порядке. Я обновил свой вопрос.

Ответ №3:

Прежде всего, я надеюсь, что вы добавили hilt-android-gradle-plugin в свой проект.

Во-вторых, в кинжале давно была проблема с этой зависимостью от наследования, но она уже исправлена с 2.28. Вы можете увидеть более подробную информацию по этой ссылке. https://github.com/google/dagger/issues/1910

Итак, для вашего случая без добавления @AndroidEntryPoint BottomSheetDialogFragment должно сработать, но вы также пытались добавить это? Так ли это работало?

Кроме того, не могли бы вы поделиться своим классом фрагментации нижней таблицы?

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

1. Да, у меня есть hilt-android-gradle-plugin это в моем проекте. Что касается проблемы с GitHub, это не помогло. Я обновил свой вопрос, включив в него свой каталог нижней таблицы.

2. Судя по журналу, это не похоже на проблему с кинжалом. Какова ваша версия kotlin и версия с рукоятью кинжала? Также, какую версию java вы используете?

3. const val kotlin_version = "1.5.31" const val hilt_version = "2.38.1" Я использую Java 11

Ответ №4:

Из документации:

Рукоять поддерживает только фрагменты, расширяющие androidx.Фрагмент.

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

1. BottomSheetDialogFragment() расширяет androidx. Фрагмент.