I've tried, to create several components. First will store main parts of app, like ViewModel's Fabric, Context, other settings. Other components -- component per screen. So, f.ex. I have FirstScreen. I've tried to create component with its ViewModel:
@Subcomponent(modules = [StoreModule::class])
@StoreScope
interface StoreComponent {
fun inject(activity: MainActivity)
fun inject(fragment: StoreFragment)
@Subcomponent.Builder
interface Builder {
fun build(): StoreComponent
}
}
So, StoreViewModel, as its dependency, StoreRepository, builds in module of StoreComponent:
@Module
abstract class StoreModule {
@Binds
@IntoMap
@ViewModelKey(StoreViewModelImpl::class)
abstract fun getStoreViewModel(viewModel: StoreViewModelImpl): ViewModel
@Binds
@StoreScope
abstract fun getStoreRepository(repository: StoreRepositoryImpl): StoreRepository
}
And this is AppComponent:
@Component(modules = [
GsonModule::class,
RetrofitModule::class,
ViewModelsFactoryModule::class,
CiceroneModule::class
])
@Singleton
interface AppComponent {
fun getStoreComponentBuilder(): StoreComponent.Builder
}
Here buildes ViewModel's Factory:
@Module
abstract class ViewModelsFactoryModule {
@Target(AnnotationTarget.FUNCTION,
AnnotationTarget.PROPERTY_GETTER,
AnnotationTarget.PROPERTY_SETTER
)
@Retention(AnnotationRetention.RUNTIME)
@MapKey
internal annotation class ViewModelKey(val value: KClass<out ViewModel>)
@Binds
abstract fun getViewModelFactory(factory: ViewModelFactory): ViewModelProvider.Factory
}
But, when I try to build project, I have en error, that dagger can't provide Map<java.lang.Class<? extends android.arch.lifecycle.ViewModel>
into ViewModelFactory-class without @Provides-annonated method.
error: [com.sagrishin.smartreader.di.components.StoreComponent.inject(com.sagrishin.smartreader.presentation.fragments.StoreFragment)] java.util.Map<java.lang.Class<? extends android.arch.lifecycle.ViewModel>,javax.inject.Provider<android.arch.lifecycle.ViewModel>> cannot be provided without an @Provides-annotated method.
public abstract interface AppComponent {
^
java.util.Map<java.lang.Class<? extends android.arch.lifecycle.ViewModel>,javax.inject.Provider<android.arch.lifecycle.ViewModel>> is injected at
com.sagrishin.smartreader.presentation.viewmodels.factory.ViewModelFactory.<init>(creators)
com.sagrishin.smartreader.presentation.viewmodels.factory.ViewModelFactory is injected at
com.sagrishin.smartreader.presentation.fragments.StoreFragment.viewModelsFactory
com.sagrishin.smartreader.presentation.fragments.StoreFragment is injected at
com.sagrishin.smartreader.di.components.StoreComponent.inject(fragment)
UPD: viewmodel factory:
typealias ViewModelsProvidersMap =
Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>
@Singleton
class ViewModelFactory : ViewModelProvider.Factory {
private val creators: ViewModelsProvidersMap
@Inject
constructor(creators: ViewModelsProvidersMap) {
this.creators = creators
}
override fun <T : ViewModel> create(modelClass: Class<T>): T {
var creator: Provider<out ViewModel>? = creators[modelClass]
if (creator == null) {
for ((key, value) in creators) {
if (modelClass.isAssignableFrom(key)) {
creator = value
break
}
}
}
if (creator == null) {
throw IllegalArgumentException("unknown model class $modelClass")
}
try {
return creator.get() as T
} catch (e: Exception) {
throw RuntimeException(e)
}
}
}