Dagger 2 将活动的视图模型注入片段

The*_*mer 3 android kotlin dagger-2

(使用 kotlin)我有一个应用程序,它使用带有 2 个片段的设置活动。我希望两者都获得与活动相同的 SettingsViewModel 实例。我认为我遗漏了一个范围界定问题。

首先,我有标准ViewModelModule

@Module
abstract class ViewModelModule {
    @Binds @IntoMap
    @ViewModelKey(SettingsViewModel::class)
    abstract fun bindSettingsViewModel(viewModel: SettingsViewModel): ViewModel

    @Binds
    abstract fun bindViewModelFactory(factory: MyViewModelFactory): ViewModelProvider.Factory
}
Run Code Online (Sandbox Code Playgroud)

我将我的活动绑定在:

@Module
abstract class AndroidBindingModule {
    @PerActivity
    @ContributesAndroidInjector(modules = [SettingsActivityModule::class])
    abstract fun contributeSettingsActivity(): SettingsActivity
}
Run Code Online (Sandbox Code Playgroud)

有了所有其他的东西,这工作得很好,并且SettingsActivity确实获得了SettingsViewModel. SettingsActivityModule添加以下内容:

@PerFragment
@ContributesAndroidInjector
abstract fun contributeMainSettingsFragment(): MainSettingsFragment

@PerFragment
@ContributesAndroidInjector
abstract fun contributeDebugSettingsFragment(): DebugSettingsFragment
Run Code Online (Sandbox Code Playgroud)

两个片段似乎都调用了注入器(我已经检查过调试器并被AndroidSupportInjection.inject(fragment)调用)。片段包括:

@Inject lateinit var mainViewModel: SettingsViewModel
Run Code Online (Sandbox Code Playgroud)

但在我的片段中,onCreate()我看到它mainViewModel仍然为空。我需要在这里做什么特别的事情来避免调用ViewModelProviders.of(activity)[SettingsViewModel::class.java]而是注入视图模型?

更新:

阅读更多内容后,我发现在片段中使用视图模型注入的正确方法是注入工厂并获取视图模型onActivityCreated

@Inject lateinit var viewModelFactory: ViewModelProvider.Factory
lateinit var mainViewModel: SettingsViewModel

override fun onActivityCreated(savedInstanceState: Bundle?) {
    super.onActivityCreated(savedInstanceState)
    mainViewModel = ViewModelProviders.of(activity, viewModelFactory)[SettingsViewModel::class.java]
}
Run Code Online (Sandbox Code Playgroud)

这是有道理的,因为我已经MyViewModelFactory绑定为ViewModelProvider.Factory并且它用@Singleton. 当我尝试编译上述代码时,出现以下错误:

Error:(6, 1) error: [dagger.android.AndroidInjector.inject(T)] java.util.Map<kotlin.reflect.KClass<? extends android.arch.lifecycle.ViewModel>,? extends javax.inject.Provider<android.arch.lifecycle.ViewModel>> cannot be provided without an @Provides-annotated method.
Run Code Online (Sandbox Code Playgroud)

似乎 Dagger 找不到由ViewModelModule. 我仍然不知道这是怎么回事。也许我的树不正确?为什么活动AndroidBindingModule能够获得 ViewModel 而不是片段?

AppComponent
  - AndroidInjectionModule
  - AndroidBindingModule
  - AppModule
    - SdkModule
    - ViewModelModule
    - GotItCardModule
    - ViewHolderSubcomponent (provides a mapping of layout ID -> ViewHolder for a factory)
Run Code Online (Sandbox Code Playgroud)

更新

我已经做了更多的挖掘......从完整的错误:

e: /home/user/workspace/Example/sdktest/build/tmp/kapt3/stubs/debug/com/example/sdktest/di/AppComponent.java:6: error: [dagger.android.AndroidInjector.inject(T)] java.util.Map<kotlin.reflect.KClass<? extends android.arch.lifecycle.ViewModel>,? extends javax.inject.Provider<android.arch.lifecycle.ViewModel>> cannot be provided without an @Provides-annotated method.
e: 

e: public abstract interface AppComponent {
e:                 ^
e:       java.util.Map<kotlin.reflect.KClass<? extends android.arch.lifecycle.ViewModel>,? extends javax.inject.Provider<android.arch.lifecycle.ViewModel>> is injected at
e:           com.example.sdktest.di.viewmodel.ExampleViewModelFactory.<init>(creators)
e:       com.example.sdktest.di.viewmodel.ExampleViewModelFactory is injected at
e:           com.example.sdktest.di.viewmodel.ViewModelModule.bindViewModelFactory(factory)
e:       android.arch.lifecycle.ViewModelProvider.Factory is injected at
e:           com.example.sdktest.ui.settings.fragment.MainSettingsFragment.viewModelFactory
e:       com.example.sdktest.ui.settings.fragment.MainSettingsFragment is injected at
e:           dagger.android.AndroidInjector.inject(arg0)
Run Code Online (Sandbox Code Playgroud)

我认为这个问题是,不知何故匕首试图与注入我的片段dagger.android.AndroidInjecton代替dagger.android.AndroidSupportInjection。仍然不确定如何修复。

The*_*mer 5

好的,所以我找到了答案,它与我的想法相去甚远。我将GithubViewModelFactory翻译成 Kotlin 包括以下构造函数:

@Singleton
class MetaverseViewModelFactory @Inject constructor(
        creators: Map<KClass<out ViewModel>, Provider<ViewModel>>
): ViewModelProvider.Factory {
    private val creators: Map<Class<out ViewModel>, Provider<ViewModel>> =
            creators.mapKeys { it.key.java }
    //...
}
Run Code Online (Sandbox Code Playgroud)

这是由于ViewModelKey在科特林能够使用KClass而不是只Class。事实证明,kapt 会处理这个问题,正确的工厂应该是这样的:

@Singleton
class MetaverseViewModelFactory @Inject constructor(
        private val creators: Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>
): ViewModelProvider.Factory {
    //...
}
Run Code Online (Sandbox Code Playgroud)

注意额外@JvmSupressWildcards的也避免Provider<ViewModel>变成Provider<? extends ViewModel>