Blogs/Technology

Reducing Intent data passing ‘putExtra’, ‘putInt’, ‘putParcelable’ Boiler plate code

Written by Murtuza Kutub
Mar 23, 2026
5 Min Read
Reducing Intent data passing ‘putExtra’, ‘putInt’, ‘putParcelable’ Boiler plate code Hero

Passing data between Activities using Intent extras often becomes repetitive and error-prone. I’ve noticed this especially when dealing with multiple data types like Parcelable, primitives, and bundles.

Managing putExtra, putInt, and putParcelable Repeatedly increases boilerplate and reduces readability.

Dart and Henson simplify this process by introducing a structured and type-safe way to handle Intent data passing.

Dart & Henson:

Dart and Henson are Android dependency injection libraries designed to simplify Intent data passing.

They eliminate the need for repetitive key-based methods like putParcelable, putInt, putString, and putBundle.

By using annotation processing, they generate type-safe code, reducing boilerplate and improving maintainability.

How to use?

The setup is straightforward and requires adding the following Gradle dependencies:

compile 'com.f2prateek.dart:dart:2.0.2'
annotationProcessor 'com.f2prateek.dart:dart-processor:2.0.2'


compile 'com.f2prateek.dart:henson:2.0.2'
annotationProcessor 'com.f2prateek.dart:henson-processor:2.0.2'

Once configured, annotation-based injection replaces manual Intent handling.

Dart & Henson:

Dart and Henson is an Android dependency Injection library to deal with Intent data passing without using cumbersome putParcelable, putInt, putString, putBundle key mechanism. It is a library created by a team of Groupon developers who presented article at DroidCon Italy 2017.

How to use?

The library setup is very simple and easy to use by just adding following Gradle dependencies:

compile 'com.f2prateek.dart:dart:2.0.2'
annotationProcessor 'com.f2prateek.dart:dart-processor:2.0.2'


compile 'com.f2prateek.dart:henson:2.0.2'
annotationProcessor 'com.f2prateek.dart:henson-processor:2.0.2'

That’s it you’re done to use Injection mechanism for Intent data sharing in your project.

Usages:

To simplify implementation, consider two activities:

Automate Your Android Workflow

We build Android Studio plugins that cut manual effort, improve speed, and make your development process seamless.

Source Activity — MainActivity.java
Initiates navigation and passes data such as User (Parcelable), fromScreen (String), and isSuccess (boolean).

Target Activity — DetailsActivity.java
Receives and displays the passed data using injection.

Implementation:

What is Dart?

Dart is an injection library that assigns Intent extras directly to fields in the target activity using annotations.

Instead of manually retrieving extras, Dart maps them automatically during activity initialization.

import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.widget.Button;
import android.widget.TextView;

import com.f2prateek.dart.Dart;
import com.f2prateek.dart.InjectExtra;

import butterknife.BindView;
import butterknife.ButterKnife;
@BindView(R.id.txt_mobile)
TextView txtMobile;
@BindView(R.id.txt_name)
TextView txtName;
@BindView(R.id.txt_email)
TextView txtEmail;
@BindView(R.id.btn_click_me)
Button btnClickMe;


@InjectExtra User user;
@Nullable
@InjectExtra String fromScreen;
@InjectExtra boolean isSuccess = false;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_details);
    ButterKnife.bind(this);

    Dart.inject(this);
}

STEP 1: Inject Dart in Target Activity

Dart.inject(this);

STEP 2: Define fields using @InjectExtra

@InjectExtra User user;

@Nullable
@InjectExtra String fromScreen;

@InjectExtra boolean isSuccess = false;

Optional Injection:

Infographic titled “Optional Injection” highlighting defining optional @InjectExtra fields, using @Nullable annotation, allowing missing intent data safely, and preventing runtime issues during injection.

Optional injection allows fields to remain unset if not provided.

Using @Nullable ensures the field does not cause runtime issues when missing.

Default Values:

Default values can be assigned directly during field declaration.

This ensures fallback behavior without additional checks during runtime.

What is Henson?

Henson handles navigation between activities while passing Intent data in a structured way.

It generates builder-style methods based on injected fields, reducing manual Intent construction.

import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.widget.Button;
import android.widget.EditText;

import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.OnClick;

public class MainActivity extends AppCompatActivity {

    @BindView(R.id.edt_mobile)
    EditText edtMobile;
    @BindView(R.id.edt_name)
    EditText edtName;
    @BindView(R.id.edt_email)
    EditText edtEmail;
    @BindView(R.id.btn_login)
    Button btnLogin;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ButterKnife.bind(this);

    }

    @OnClick(R.id.btn_login)
    public void onLoginClicked(){
        Intent intent = Henson.with(MainActivity.this)
                .gotoDetailsActivity()
                .build();
   }
}

Yeah! Now Henson setup is done. Click BuildRebuild Project

Chill! I know compiler throws Compilation error in .build() method as follows

Messages gradle build

The reason behind this because we haven’t passed any @InjectExtra annotation methods which are created in DetailsActivity.java

@InjectExtra User user;
@Nullable
@InjectExtra String fromScreen;
@InjectExtra boolean isSuccess = false;

Cool. Let’s add all the parameters in the Intent as

import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.widget.Button;
import android.widget.EditText;


import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.OnClick;

public class MainActivity extends AppCompatActivity {

    @BindView(R.id.edt_mobile)
    EditText edtMobile;
    @BindView(R.id.edt_name)
    EditText edtName;
    @BindView(R.id.edt_email)
    EditText edtEmail;
    @BindView(R.id.btn_login)
    Button btnLogin;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ButterKnife.bind(this);

    }


    @OnClick(R.id.btn_login)
    public void onLoginClicked(){

        User user = new User();
        user.setEmail(edtEmail.getText().toString());
        user.setMobile(edtMobile.getText().toString());
        user.setName(edtName.getText().toString());

        Intent intent = Henson.with(MainActivity.this)
                .gotoDetailsActivity()
                .isSuccess(true)
                .user(user)
                .build();

        startActivity(intent);

    }
}

Note: We haven’t passed fromScreen method since it is optional (By default it will be ‘null’).

What is .gotoDetailsActivity()?

I know everyone would be wondering from where this .gotoDetailsActivity() comes into the picture? Cool we have a answer for the same.

goto (Henson generated appending string to start an Activity) + DetailsActivity (Our own TargetActivity).

Yeah! We’re almost done. Let’s play with the intents in the TargetActivity (DetailsActivity.java)

import android.content.Intent;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;

import com.f2prateek.dart.Dart;
import com.f2prateek.dart.InjectExtra;

import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.OnClick;

public class DetailsActivity extends AppCompatActivity {

    @BindView(R.id.txt_mobile)
    TextView txtMobile;
    @BindView(R.id.txt_name)
    TextView txtName;
    @BindView(R.id.txt_email)
    TextView txtEmail;
    @BindView(R.id.btn_click_me)
    Button btnClickMe;


    @InjectExtra User user;

    @Nullable
    @InjectExtra String fromScreen;

    @InjectExtra boolean isSuccess = false;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_details);
        ButterKnife.bind(this);

        Dart.inject(this);

        updateFields();
    }

    private void updateFields() {

        txtMobile.setText(user.getMobile());

        txtEmail.setText(user.getEmail());

        txtName.setText(user.getName());

    }

    @OnClick(R.id.btn_click_me)
    public void onClickMeClicked(){

        if(isSuccess){
            Toast.makeText(DetailsActivity.this, "Dart and Henson is really working!", Toast.LENGTH_SHORT).show();
        }
    }

}

FAQ

What problem do Dart and Henson solve?

They reduce boilerplate code in Intent data passing by replacing manual key-based methods with annotation-based injection.

Automate Your Android Workflow

We build Android Studio plugins that cut manual effort, improve speed, and make your development process seamless.

How does Dart work in Android?

Dart uses annotation processing to inject Intent extras directly into fields in the target activity.

What is Henson used for?

Henson simplifies navigation between activities by generating type-safe builder methods for passing data.

Are Dart and Henson type-safe?

Yes. They generate compile-time safe code, reducing runtime errors caused by missing or incorrect Intent data.

Can optional data be passed using Dart?

Yes. Fields annotated with @Nullable can be omitted without causing issues.

Author-Murtuza Kutub
Murtuza Kutub

A product development and growth expert, helping founders and startups build and grow their products at lightning speed with a track record of success. Apart from work, I love to Network & Travel.

Share this article

Phone

Next for you

8 Best GraphQL Libraries for Node.js in 2025 Cover

Technology

Jan 29, 20268 min read

8 Best GraphQL Libraries for Node.js in 2025

Why do some GraphQL APIs respond in milliseconds while others take seconds? The difference often comes down to choosing the right GraphQL library for Node.js. According to npm trends, Apollo Server Express alone sees over 800,000 weekly downloads, proving that developers need reliable tools to build production-ready GraphQL servers. The truth is, building GraphQL APIs in Node.js has never been easier, but picking the wrong library can slow down your entire application. Modern web applications d

I Tested 9 React Native Animation Libraries (Here’s What Works) Cover

Technology

Feb 10, 202614 min read

I Tested 9 React Native Animation Libraries (Here’s What Works)

Why do some mobile apps feel smooth while others feel clunky? I’ve noticed the difference is usually animations under load, especially during scrolling, navigation, and gesture-heavy screens. Google research shows 53% of mobile site visits are abandoned if pages take longer than three seconds to load, and the same performance expectations carry over to mobile apps. The truth is, smooth animations in React Native apps are no longer a luxury; they’re a must-have for a modern, engaging user experi

9 Critical Practices for Secure Web Application Development Cover

Technology

Jan 29, 20267 min read

9 Critical Practices for Secure Web Application Development

In 2026, developing modern web applications requires a balance between speed and security. Product strategy often pressures development teams to move fast, and ignoring application security can cause catastrophic results. For example, post-credential-based attacks have caused over $5 billion in losses. Security vulnerabilities in web applications are not just technical security problems; they are a business risk. The truth is that security incidents happen when web developers think about web se