Kotlin, Dagger2, MVVM и ContributesAndroidInjector - PullRequest
3 голосов
/ 01 июня 2019

У нас возникла проблема с Dagger2, когда модули, указанные с помощью ContributesAndroidInjector, не распознаются во время компиляции.

// This doesn't work
@Module
abstract class ActivityBuildersModule {

    @ContributesAndroidInjector(
            modules = [
                FragmentBuildersModule::class,
                FeatureViewModelModule::class,
            ]
    )
    abstract fun contributeMainActivity(): MainActivity
}

Это наш модуль, в результате которого возникает следующая ошибка:

javax.inject.Provider<androidx.lifecycle.ViewModel>> cannot be provided without an @Provides-annotated method.

Если мы поднимаем объявление до уровня класса, все работает как ожидалось:

// This works
@Module(includes = [
    FeatureViewModelModule::class,
])
abstract class ActivityBuildersModule {

    @ContributesAndroidInjector(
            modules = [
                FragmentBuildersModule::class,
            ]
    )
    abstract fun contributeMainActivity(): MainActivity
}

Обратите внимание, что эта проблема не является исключительной для FeatureViewModelModule.Любой модуль, который обеспечивает некоторую зависимость, не будет работать, если поместить его в @ ContributesAndroidInjector.

Что еще интересно, есть ли проект на GitHub, который, кажется, делает то же самоебез проблем.Единственное различие, которое я вижу между проектами, заключается в том, что наши проекты находятся в Котлине, а их - в Java.

Некоторая дополнительная информация:

class MyApplication : DaggerApplication(){

    override fun applicationInjector(): AndroidInjector<out DaggerApplication> {
        return DaggerAppComponent.builder().application(this).build()
    }
}
@Singleton
@Component(
        modules = [
            AndroidInjectionModule::class,
            ActivityBuildersModule::class,
            AppModule::class]
)
interface AppComponent : AndroidInjector<MyApplication> {
    @Component.Builder
    interface Builder {
        @BindsInstance
        fun application(application: Application): Builder

        fun build(): AppComponent
    }
}
@Singleton
class ViewModelFactory @Inject constructor(
    private val creators: Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>
) : ViewModelProvider.Factory {
    override fun <T : ViewModel> create(modelClass: Class<T>): T {
        val creator = creators[modelClass] ?: creators.entries.firstOrNull {
            modelClass.isAssignableFrom(it.key)
        }?.value ?: throw IllegalArgumentException("unknown model class $modelClass")
        try {
            @Suppress("UNCHECKED_CAST")
            return creator.get() as T
        } catch (e: Exception) {
            throw RuntimeException(e)
        }

    }
}
@Suppress("unused")
@Module
abstract class FeatureViewModelModule {
    @Binds
    @IntoMap
    @ViewModelKey(FeatureViewModel::class)
    abstract fun bindCompanyViewModel(featureViewModel: FeatureViewModel): ViewModel
}

Кто-нибудь есть какие-либо идеи, что может происходить?

...