Кинжал: не могу предоставить картубез @ Предоставляет аннотированный метод - PullRequest
0 голосов
/ 11 декабря 2018

Я пытался создать несколько компонентов.Сначала будут храниться основные части приложения, такие как ViewModel's Fabric, Context, другие настройки.Другие компоненты - компонент на экран.Так, напримерУ меня есть FirstScreen.Я пытался создать компонент с его ViewModel:

@Subcomponent(modules = [StoreModule::class])
@StoreScope
interface StoreComponent {

    fun inject(activity: MainActivity)

    fun inject(fragment: StoreFragment)

    @Subcomponent.Builder
    interface Builder {

        fun build(): StoreComponent

    }

}

Итак, StoreViewModel, как его зависимость, StoreRepository, строит в модуле StoreComponent:

@Module
abstract class StoreModule {

    @Binds
    @IntoMap
    @ViewModelKey(StoreViewModelImpl::class)
    abstract fun getStoreViewModel(viewModel: StoreViewModelImpl): ViewModel

    @Binds
    @StoreScope
    abstract fun getStoreRepository(repository: StoreRepositoryImpl): StoreRepository

}

И это AppComponent:

@Component(modules = [
    GsonModule::class,
    RetrofitModule::class,
    ViewModelsFactoryModule::class,
    CiceroneModule::class
])
@Singleton
interface AppComponent {

    fun getStoreComponentBuilder(): StoreComponent.Builder

}

Здесь строит фабрику ViewModel:

@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

}

Но, когда я пытаюсь построить проект, у меня возникает ошибка, из-за которой кинжал не может предоставить Map<java.lang.Class<? extends android.arch.lifecycle.ViewModel> в ViewModelFactory-классебез @ Предоставляет метод, анонированный.

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: фабрика представления модели:

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)
        }

    }
}

Ответы [ 3 ]

0 голосов
/ 23 мая 2019

Вам не нужно переключаться на Java, просто используйте kotlin 1.3.31, чтобы исправить это.https://github.com/google/dagger/issues/1478#issuecomment-486712176

включает следующую аннотацию Jvm

Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>
0 голосов
/ 04 июля 2019
No need to use @JvmSuppressWildcards. As the day of my Post my Kotlin version
 is : '1.3.40'

 Use dagger dependency in build.gradle

 apply plugin: 'kotlin-kapt'

 //Dagger dependencies
implementation 'com.google.dagger:dagger:2.22.1'
kapt 'com.google.dagger:dagger-compiler:2.22.1'

implementation 'com.google.dagger:dagger-android:2.22'
implementation 'com.google.dagger:dagger-android-support:2.22' // if you use the support libraries
kapt 'com.google.dagger:dagger-android-processor:2.22'


And for Java use the following dagger dependency

 //Dagger dependencies
implementation 'com.google.dagger:dagger:2.22.1'
annotationProcessor 'com.google.dagger:dagger-compiler:2.22.1'

implementation 'com.google.dagger:dagger-android:2.22'
implementation 'com.google.dagger:dagger-android-support:2.22' // if you use the support libraries
annotationProcessor 'com.google.dagger:dagger-android-processor:2.22'
0 голосов
/ 16 апреля 2019

Я исправил ту же проблему, что и у меня, переключив ViewModelFactory.kt и ViewModelsFactoryModule.kt на java.

ViewModelFactory.java

@Singleton
public class ViewModelFactory implements ViewModelProvider.Factory {
    private final Map<Class<? extends ViewModel>, Provider<ViewModel>> creators;

    @Inject
    ViewModelFactory(Map<Class<? extends ViewModel>, Provider<ViewModel>> creators) {
        this.creators = creators;
    }

    @NonNull
    @SuppressWarnings("unchecked")
    @Override
    public <T extends ViewModel> T create(@NonNull Class<T> modelClass) {
        Provider<? extends ViewModel> creator = creators.get(modelClass);
        if (creator == null) {
            for (Map.Entry<Class<? extends ViewModel>, Provider<ViewModel>> entry : creators.entrySet()) {
                if (modelClass.isAssignableFrom(entry.getKey())) {
                    creator = entry.getValue();
                    break;
                }
            }
        }
        if (creator == null) {
            throw new IllegalArgumentException("unknown model class " + modelClass);
        }
        try {
            return (T) creator.get();
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
}

ViewModelsFactoryModule.java

@Module
abstract class ViewModelModule {

    @NonNull
    @Binds
    @IntoMap
    @ViewModelKey(StoreViewModelImpl.class)
    abstract ViewModel bindLauncherViewModel(StoreViewModelImpl viewModel);

    @NonNull
    @Binds
    abstract ViewModelProvider.Factory bindViewModelFactory(ViewModelFactory viewModelFactory);
}
...