Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How do you organise your Dagger 2 modules and components? [closed]

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?

like image 423
ivacf Avatar asked Aug 05 '15 13:08

ivacf


People also ask

Can dagger have multiple components?

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.

What is module in dagger?

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.

What is subcomponent in dagger?

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.


1 Answers

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());             }         }     } } 
like image 139
EpicPandaForce Avatar answered Sep 18 '22 21:09

EpicPandaForce