Do you have a specific package where you put all the Dagger related classes?
Or do you put them next to the relevant class they inject, e.g. if you have an MainActivityModule
and MainActivityComponent
, you put them in the same package as your MainActivity
.
Also, I've seen quite a few people defining components as inner classes, e.g. an ApplicationComponent
that is defined inside the Application
class. Do you think this is a good practice?
Reusability is one of the many attributes that are said to contribute to a high quality code base. Dagger is in many aspects a tool which helps a lot with writing reusable code.
Modules are a way of telling Dagger how to provide dependencies from the dependency graph. These are typically high level dependencies that you aren't already contributing to the dependency graph through the @Inject constructor annotation we discussed in our previous article.
In Dagger 2, component is the main container-like object that binds all the dependencies (or it's factory). Subcomponent are components that is like an extension to its parent component. It can be used for. Partition the dependencies into different compartments.
EDIT: Let me start out with the fact that this is close to the truth here, but this is an antipattern as described by Martin Fowler's Data Domain Presentation Layering
article HERE (CLICK THE LINK!), which specifies that you shouldn't have a MapperModule
and a PresenterModule
, you should have a GalleryModule
and a SomeFeatureModule
which has all the mappers, presenters etc. in it.
The smart route to go about it is to use component dependencies to subscope your original singleton component for each feature you have. This what I described is the "full-stack" layering, separation by features.
The one written down below is the "anti-pattern", where you cut your application's top level modules into "layers". It has numerous disadvantages to do so. Don't do it. But you can read it and learn what not to do.
ORIGINAL TEXT:
Normally, you'd use a single Component
like an ApplicationComponent
to contain all singleton dependencies that you use throughout the app as long as the entire application exists. You would instantiate this in your Application class, and make this accessible from elsewhere.
Project structure for me currently is:
+ injection |- components |-- ApplicationComponent.java |- modules |- data |-- DbMapperModule.java |-- ... |- domain |-- InteractorModule.java |-- ... |- presentation |-- ... |- utils |-- ... |- scope |- subcomponents |- data |-- ... |- domain |-- DbMapperComponent.java |-- ... |- presentation |-- ... |- utils |-- ... |-- AppContextComponent.java |-- AppDataComponent.java |-- AppDomainComponent.java |-- AppPresentationComponent.java |-- AppUtilsComponent.java
For example, mine is like this:
public enum Injector { INSTANCE; private ApplicationComponent applicationComponent; private Injector() { } public ApplicationComponent getApplicationComponent() { return applicationComponent; } ApplicationComponent initializeApplicationComponent(CustomApplication customApplication) { AppContextModule appContextModule = new AppContextModule(customApplication); RealmModule realmModule = new RealmModule(customApplication.getRealmHolder()); applicationComponent = DaggerApplicationComponent.builder() .appContextModule(appContextModule) .realmModule(realmModule) .build(); return applicationComponent; } }
And you need an ApplicationComponent
that can inject into whatever package-protected fields of whatever class you want to field-inject to.
@Singleton @Component(modules = { AppContextModule.class, DbMapperModule.class, DbTaskModule.class, RealmModule.class, RepositoryModule.class, InteractorModule.class, ManagerModule.class, ServiceModule.class, PresenterModule.class, JobManagerModule.class, XmlPersisterModule.class }) public interface ApplicationComponent extends AppContextComponent, AppDataComponent, AppDomainComponent, AppUtilsComponent, AppPresentationComponent { void inject(CustomApplication customApplication); void inject(DashboardActivity dashboardActivity); ... }
For me, AppContextComponent
would be a @Subcomponent
, but that is not actually what it means. Those are just a way to create a subscope, and not a way to cut your component into smaller parts. So the interface I inherit is actually just a normal interface
with provision methods. Same for the others.
public interface AppContextComponent { CustomApplication customApplication(); Context applicationContext(); AppConfig appConfig(); PackageManager packageManager(); AlarmManager alarmManager(); }
Component dependencies (which allows you to subscope, just like subcomponents) don't allow multiple scoped components, which also means your modules would be unscoped. This is due to how you cannot inherit from multiple scopes, just like you can't inherit from multiple classes in Java.
Unscoped providers make it so that the module does not retain a single instance but a new one on every inject call. To get scoped dependencies, you need to provide the scope on the module provider methods too.
@Module public class InteractorModule { @Provides @Singleton public LeftNavigationDrawerInteractor leftNavigationDrawerInteractor() { return new LeftNavigationDrawerInteractorImpl(); } ... }
In an application, if you use Singleton components everywhere, you won't need more components, unless you create subscopes. If you want, you can even consider making your modules a complete data provider for your views and presenters.
@Component(dependencies = {ApplicationComponent.class}, modules = {DetailActivityModule.class}) @ActivityScope public interface DetailActivityComponent extends ApplicationComponent { DataObject data(); void inject(DetailActivity detailActivity); } @Module public class DetailActivityModule { private String parameter; public DetailActivityModule(String parameter) { this.parameter = parameter; } @Provides public DataObject data(RealmHolder realmHolder) { Realm realm = realmHolder.getRealm(); return realm.where(DataObject.class).equalTo("parameter", parameter).findFirst(); } }
Subscoping allows you to have multiple instances of your presenter, which can then store state. This makes sense in for example Mortar/Flow, where each screen has its own "path", and each path has its own component - to provide the data as a "blueprint".
public class FirstPath extends BasePath { public static final String TAG = " FirstPath"; public final int parameter; public FirstPath(int parameter) { this.parameter = parameter; } //... @Override public int getLayout() { return R.layout.path_first; } @Override public FirstViewComponent createComponent() { FirstPath.FirstViewComponent firstViewComponent = DaggerFirstPath_FirstViewComponent.builder() .applicationComponent(InjectorService.obtain()) .firstViewModule(new FirstPath.FirstViewModule(parameter)) .build(); return firstViewComponent; } @Override public String getScopeName() { return TAG + "_" + parameter; } @ViewScope //needed @Component(dependencies = {ApplicationComponent.class}, modules = {FirstViewModule.class}) public interface FirstViewComponent extends ApplicationComponent { String data(); FirstViewPresenter firstViewPresenter(); void inject(FirstView firstView); void inject(FirstViewPresenter firstViewPresenter); } @Module public static class FirstViewModule { private int parameter; public FirstViewModule(int parameter) { this.parameter = parameter; } @Provides public String data(Context context) { return context.getString(parameter); } @Provides @ViewScope //needed to preserve scope public FirstViewPresenter firstViewPresenter() { return new FirstViewPresenter(); } } public static class FirstViewPresenter extends ViewPresenter<FirstView> { public static final String TAG = FirstViewPresenter.class.getSimpleName(); @Inject String data; public FirstViewPresenter() { Log.d(TAG, "First View Presenter created: " + toString()); } @Override protected void onEnterScope(MortarScope scope) { super.onEnterScope(scope); FirstViewComponent firstViewComponent = scope.getService(DaggerService.TAG); firstViewComponent.inject(this); Log.d(TAG, "Data [" + data + "] and other objects injected to first presenter."); } @Override protected void onSave(Bundle outState) { super.onSave(outState); FirstView firstView = getView(); outState.putString("input", firstView.getInput()); } @Override protected void onLoad(Bundle savedInstanceState) { super.onLoad(savedInstanceState); if(!hasView()) { return; } FirstView firstView = getView(); if(savedInstanceState != null) { //needed check firstView.setInput(savedInstanceState.getString("input")); } } public void goToNextActivity() { FirstPath firstPath = Path.get(getView().getContext()); if(firstPath.parameter != R.string.hello_world) { Flow.get(getView()).set(new FirstPath(R.string.hello_world)); } else { Flow.get(getView()).set(new SecondPath()); } } } }
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With