Ошибка [Dagger / MissingBinding] androidx.lifecycle.ViewModelProvider.Factory не может быть предоставлена ​​без аннотируемого метода @ Provides - PullRequest
0 голосов
/ 24 июня 2019

Возникла проблема при реализации MultiBinding с использованием кинжала 2.2.Я использую кинжал с архитектурой MVVM.Я внедрил конструктор ViewModelProvideFactory и связал зависимость от модуля.

Я следовал учебному пособию Митча с YouTube

https://www.youtube.com/watch?v=DToD1W9WdsE&list=PLgCYzUzKIBE8AOAspC3DHoBNZIBHbIOsC&index=13

Я искал по этимссылки на решения, но все еще сталкиваются с той же проблемой.

Dagger2: ViewModel не может быть предоставлен без аннотируемого метода @ Provides

Dagger / MissingBinding java.util.Map, Поставщик > не может быть предоставлен без аннотированного метода @ Provides

https://github.com/google/dagger/issues/1478

Фрагмент кода

ViewModelKey

@MapKey
@Documented
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ViewModelKey {
    Class<? extends ViewModel> value();
}

ViewModelFactoryModule

   /**
 * ViewModelFactoryModule responsible for providing [ViewModelProviderFactory]
 *
 * Annotated with Module to tell dagger it is a module to provide [ViewModelProviderFactory]
 *
 * Annotated with bind annotation to efficiently provide dependencies similar to provides annotation
 */
@Module
abstract class ViewModelFactoryModule {

    @Binds
    abstract fun bindViewModelFactory(viewModelFactory: ViewModelProviderFactory) : ViewModelProvider.Factory
}

ViewModelProviderFactory

@Singleton
class ViewModelProviderFactory @Inject
constructor(private val creators: Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>) :
    ViewModelProvider.Factory {
override fun <T : ViewModel> create(modelClass: Class<T>): T {
    var creator: Provider<out ViewModel>? = creators[modelClass]
    if (creator == null) { // if the viewmodel has not been created

        // loop through the allowable keys (aka allowed classes with the @ViewModelKey)
        for ((key, value) in creators) {

            // if it's allowed, set the Provider<ViewModel>
            if (modelClass.isAssignableFrom(key)) {
                creator = value
                break
            }
        }
    }

    // if this is not one of the allowed keys, throw exception
    if (creator == null) {
        throw IllegalArgumentException("unknown model class $modelClass")
    }

    // return the Provider
    try {
        return creator.get() as T
    } catch (e: Exception) {
        throw RuntimeException(e)
    }

}

    companion object {

        private val TAG = "ViewModelProviderFactor"
    }
}

StackTrace

> Task :app:kaptDebugKotlin FAILED
e: /Users/fazal/Documents/fazal/demo/AdvanceDagger2/app/build/tmp/kapt3/stubs/debug/com/example/advancedagger2/di/AppComponent.java:22: error: [Dagger/MissingBinding] java.util.Map<java.lang.Class<? extends androidx.lifecycle.ViewModel>,javax.inject.Provider<androidx.lifecycle.ViewModel>> cannot be provided without an @Provides-annotated method.
public abstract interface AppComponent extends dagger.android.AndroidInjector<com.example.advancedagger2.BaseApplication> {
                ^
      java.util.Map<java.lang.Class<? extends androidx.lifecycle.ViewModel>,javax.inject.Provider<androidx.lifecycle.ViewModel>> is injected at
          com.example.advancedagger2.viewmodel.ViewModelProviderFactory(viewModelsMap)
      com.example.advancedagger2.viewmodel.ViewModelProviderFactory is injected at
          com.example.advancedagger2.ui.AuthActivity.viewModelFactory
      com.example.advancedagger2.ui.AuthActivity is injected at
          dagger.android.AndroidInjector.inject(T) [com.example.advancedagger2.di.AppComponent → com.example.advancedagger2.di.ActivityBuilderModule_ContributeAuthActivity.AuthActivitySubcomponent]

Я понизил версию Kotlin, но все еще сталкиваюсь с той же проблемой.Подскажите что, я делаю не так?

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

Я также предоставляю ViewModel через AuthViewModel, которая находится в области действия.Когда действие уничтожает его компонент, а зависимости также уничтожаются.

AuthViewModelModule

@Module
abstract class AuthViewModelModule {

    /**
     * Binds the auth view model dependency with [ViewModelKey] to group similar [ViewModel]
     *
     * Under the hood it is providing [com.example.advancedagger2.viewmodel.AuthViewModel]
     */
    @Binds
    @IntoMap
    @ViewModelKey(AuthViewModel::class)
    abstract fun bindAuthViewModel(authViewModel: AuthViewModel) : ViewModel
}

ActivityBuilderModule

/**
 * This Class {@linkplain ActivityBuilderModule} is responsible for for android injection
 * for the activity with in the application to avoid the seprate injection in each activity
 *
 * {@linkplain dagger.android.AndroidInjection#inject(Activity)}
 *
 * {@link com.example.advancedagger2.viewmodel.AuthViewModel} can be access from Auth Activity
 * only so it is the concept of sub-modules
 *
 */
@Module
public abstract class ActivityBuilderModule {

    @ContributesAndroidInjector(
            modules = AuthViewModelModule.class
    )
    abstract AuthActivity contributeAuthActivity();
}

Ответы [ 2 ]

2 голосов
/ 25 июня 2019

По комментарию я решил вопрос

Ваш завод, вероятно, не должен быть @Singleton. Это легкий, не несет никакого состояния, и ваши видовые модели, вероятно, связаны в область действия и не доступна в @Singleton в любом случае

Я пометил фабрику @Singleton, которая недоступна в области действия. Я только что удалил аннотацию @Singleton. Все работает как положено

0 голосов
/ 24 июня 2019

Вам необходимо добавить аннотированный метод для моделей представлений, см. Ниже.

@SuppressWarnings("unused")
@Binds
@IntoMap
@ViewModelKey(value = SomeViewModel.class)
abstract ViewModel bindSomeViewModel(SomeViewModel viewModel);


@SuppressWarnings("unused")
@Binds
abstract ViewModelProvider.Factory bindViewModelFactory(ViewModelFactory viewModelFactory);
...