Кинжал не может внедрить ViewModel с KClass - PullRequest
0 голосов
/ 24 апреля 2019

Я перевожу свой проект Java в kotlin, но у меня возникли некоторые сомнения по поводу KClass и Class

Вот мой BaseActivity

abstract class BaseActivity<DB : ViewDataBinding, VM : BaseViewModel> : DaggerAppCompatActivity() {

    private lateinit var mCustomDialog: CustomDialog
    private lateinit var mViewDataBinding: DB
    private lateinit var mViewModel : VM

    @Inject
    lateinit var viewModelFactory: ViewModelFactory

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        // Set Custom Dialog
        mCustomDialog = CustomDialog(this, R.style.LoadingDialogStyle)
        // Set ViewModel
        mViewModel = ViewModelProviders.of(this, viewModelFactory).get(getViewModelClass().java)
        // Set DataBinding
        mViewDataBinding = DataBindingUtil.setContentView(this, getLayoutId())
        mViewDataBinding.lifecycleOwner = this
        mViewDataBinding.setVariable(getBindingVariable(), mViewModel)
        mViewDataBinding.executePendingBindings()
        // Initialize UI
        prepareView(savedInstanceState)
    }


    @LayoutRes
    abstract fun getLayoutId(): Int


    protected abstract fun getViewModelClass(): KClass<VM>


    abstract fun getBindingVariable(): Int


    fun getViewModel(): VM {
        return mViewModel
    }


    fun getViewDataBinding() : DB {
        return mViewDataBinding
    }

Я использую protected abstract fun getViewModelClass(): KClass<VM> функция для инициализации ViewModel класса в функции ниже

ViewModelProviders.of(this, viewModelFactory).get(getViewModelClass().java)

Я использую ViewModel таким образом в действиях

class SplashActivity : BaseActivity<ActivitySplashBinding, SplashViewModel>() {

    override fun getViewModelClass(): KClass<SplashViewModel> {
        return SplashViewModel::class
    }

    override fun getLayoutId(): Int {
        return R.layout.activity_splash
    }

    override fun getBindingVariable(): Int {
        return BR.vm
    }

    override fun prepareView(savedInstanceState: Bundle?) {
        getViewModel().testLog()
    }

}

Но когда я запускаю проект, я получаю эту ошибку

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.example.MyApp> {
                ^
      java.util.Map<java.lang.Class<? extends androidx.lifecycle.ViewModel>,javax.inject.Provider<androidx.lifecycle.ViewModel>> is injected at
          com.example.example.utils.ViewModelFactory(viewModels)
      com.example.example.utils.ViewModelFactory is injected at
          com.example.example.base.BaseActivity.viewModelFactory
      com.example.example.ui.splash.SplashActivity is injected at
          dagger.android.AndroidInjector.inject(T) [com.example.example.di.AppComponent ? com.example.example.di.ActivityBindingsModule_SplashActivityInjector$app_debug.SplashActivitySubcomponent]

Итак, я провел небольшое исследование и выяснил, что в моем ViewModelKey

это примерно KClass Вот ViewModelKey

@Target(AnnotationTarget.FUNCTION, AnnotationTarget.PROPERTY_GETTER, AnnotationTarget.PROPERTY_SETTER)
@Retention(AnnotationRetention.RUNTIME)
@MapKey
annotation class ViewModelKey(val value: KClass<out ViewModel>)

Если я это сделаюне меняйте мой код на Kotlin и используйте старый класс Java, как этот, он работает правильно

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

Это мой ViewModelFactory класс

@Suppress("UNCHECKED_CAST")
class ViewModelFactory @Inject
constructor(private val viewModels: MutableMap<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>) : ViewModelProvider.Factory {

    override fun <T : ViewModel> create(modelClass: Class<T>): T {
        val creator = viewModels[modelClass]
            ?: viewModels.asIterable().firstOrNull { modelClass.isAssignableFrom(it.key) }?.value
            ?: throw IllegalArgumentException("unknown model class $modelClass")
        return try {
            creator.get() as T
        } catch (e: Exception) {
            throw RuntimeException(e)
        }
    }
}

Мой SplashActivityModule

@Module
abstract class SplashActivityModule {

    @Binds
    @IntoMap
    @ViewModelKey(SplashViewModel::class)
    internal abstract fun provideSplashViewModel(splashViewModel: SplashViewModel) : ViewModel

}

Так как я могу правильно использовать ViewModelKey с Kotlin и какова основная причина этой ошибки, любая помощь будет оценена

1 Ответ

0 голосов
/ 25 апреля 2019

Ваш ViewModelKey будет похож на

@MustBeDocumented
@kotlin.annotation.Target(AnnotationTarget.FUNCTION, AnnotationTarget.PROPERTY_GETTER, AnnotationTarget.PROPERTY_SETTER)
@kotlin.annotation.Retention(AnnotationRetention.RUNTIME)
@MapKey
internal annotation class ViewModelKey(val value: KClass<out ViewModel>)
...