I have simplified my application to get the root of the problem and here is the simplified version. I'm implementing Dagger 2 using following configuration:
AppComponent
@Component(modules = [
AndroidSupportInjectionModule::class,
ActivityBindingModule::class
])
interface AppComponent: AndroidInjector<MyApp> {
@Component.Builder
interface Builder{
@BindsInstance
fun application(application: Application): Builder
fun build(): AppComponent
}
}
ActivityBindingModule
@Module
abstract class ActivityBindingModule {
@ContributesAndroidInjector
abstract fun mainActivity(): MainActivity
@Module
companion object{
@JvmStatic
@Provides
fun provideString(mainActivity: MainActivity): String{
return "Tent"
}
}
}
MainActivity
class MainActivity : DaggerAppCompatActivity() {
@Inject
lateinit var string: String
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
println("meco simplest ${string}")
}
}
When I run the application I get the following error. What I don't understand is ContributesAndroidInjector
is already providing an instace of MainActivity
why Dagger still complains about it.
MainActivity cannot be provided without an @Inject constructor or an @Provides-annotated method
EDIT for @yavor
Keep all classes as is and separate ActivityBindingModule implementation into two classes. Now you can see that instance of the MainActivity
is provided and Dagger is not complaining about it.
ActivityBindingModule
@Module
abstract class ActivityBindingModule {
@ContributesAndroidInjector(modulese [StringProviderModule::class])
abstract fun mainActivity(): MainActivity
}
StringProviderModule
@Module
class StringProviderModule {
@Module
companion object{
@JvmStatic
@Provides
fun provideString(mainActivity: MainActivity): String{
return "Tent"
}
}
}
What I don't understand is ContributesAndroidInjector is already providing an instace of MainActivity why Dagger still complains about it.
ContributesAndroidInjector in docs says:
Generates an {@link AndroidInjector} for the return type of this method. The injector is implemented with a {@link dagger.Subcomponent} and will be a child of the {@link dagger.Module}'s component.
So it does not provide MainActivity.
Why do you need it actually at all? I see that you are passing it as parameter to the function:
fun provideString(mainActivity: MainActivity)
but do you really need it there? In general you should inject dependencies in MainActivity. MainActivity should use them. If both(MainActivity and the string) they know about each other it is first - not a good design, and second: you are close to create cyclic dependencies which Dagger 2 does not support and throws exception.
You probably forgot to inject your application in
MyApp
. You should have something like this (you might need to modify it a bit to fit yourAppComponent
:Also, Dagger is actually providing your
MainActivity
through your@ContributesAndroidInjector
annotated method but that's not what you're injecting.You're injecting a
string
so Dagger is using yourprovideString
method. Since this method requires aMainActivity
instance to work, Dagger is looking for such a method annotated with@Provides
. You don't have any and Dagger won't look at your@ContributesAndroidInjector
method since it does not have any reasons to do so.If you want it to work, you actually have to define your
provideString
method in a separate module and install it inside your@ContributesAndroidInjector
: