ActivityBuilder


Source link: https://github.com/douo/ActivityBuilder

ActivityBuilder

????

ActivityBuilder is a annotation base library using builder pattern to make inner activity communication more easier.

Through ActivityBuilder you can use one line of code to deliver parameters, start the Activity and handle the result:

EditorActivityBuilder.create(this)

  .hint("say something!")

  .forContent(text -> System.out.println(text))

  .start() 

How to use

implementation 'info.dourok.builder:activity-builder:0.1.65' annotationProcessor 'info.dourok.builder:activity-builder-compiler:0.1.65' 

Using ActivityBuilder need lambda expression supported?

android {

...
compileOptions {

  sourceCompatibility JavaVersion.VERSION_1_8
  targetCompatibility JavaVersion.VERSION_1_8

}
 
}
 

more detail in Use Java 8 language features | Android Studio

or use retrolambda.

Example

Assume we need to start a Activity named EditorActivity to capturing user input, and pass a string parameter as hint.

In mostly android way:

private static final int REQUEST_SOME_TEXT = 0x2;
 private void requestSomeTextNormalWay() {

  findViewById(R.id.fab).setOnClickListener(

view -> {

  Intent intent = new Intent(this, EditorActivity.class);

  intent.putExtra("hint", "say something");

  startActivityForResult(intent, REQUEST_SOME_TEXT);

}

  );

}

 @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) {

  switch (requestCode) {

 case REQUEST_SOME_TEXT:

if (resultCode == EditorActivityHelper.RESULT_CONTENT) {

  String text = data.getStringExtra("content");

  System.out.println(text);

}

  
}

}
 

Using ActivityBuilder you can solve it by one line of code :)

private void requestSomeText() {

  findViewById(R.id.fab).setOnClickListener(

view ->

 EditorActivityBuilder.create(this)

  .hint("say something!")

  .forContent(System.out::println)

  .start()
  );

}
 

The most thing you need to do is add some annotation, and ActivityBuilder will take care the rest.

@Builder @Result(name = "content", parameters = {
 @ResultParameter(name = "content", type = String.class) 
}
) public class EditorActivity extends AppCompatActivity {

@BuilderParameter String hint;
... 
}
 

Another Example: Using exist activity

The following example starts the system camera app and get a photo, converts the Builder to Intent with the asIntent, configures some parameters and then converts it back to Builder via asBuilder and sets the callback and starts the Activity. tmpFile is declared as a local variable capturing by lambda to avoid declared as a class variables. Full code See: CameraActivity.java

private void takePhoto() {

  Intent intentPhoto = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

  ComponentName componentName = intentPhoto.resolveActivity(getPackageManager());

  if (componentName != null) {

 // Use lambda expressions to capture local variables, avoiding the use of tmpFile as a class variable.

 File tmpFile = getTempFile(FileType.IMG);

 if (tmpFile != null) {

// start camera

// Note that BuilderUtil is only generated by using @Builder annotations

// you can use BaseActivityBuilder.create instead

BuilderUtil.createBuilder(this, intentPhoto)

 .asIntent() // convert to intent

 .setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION)

 .putExtra(MediaStore.EXTRA_OUTPUT, getUri(componentName, tmpFile))

 .asBuilder() // convert to builder

 .forOk((context, intent) -> context.showPicture(tmpFile))

 .start();

 
}

  
}

}
 private void showPicture(File file) {

  ImageView imageView = findViewById(R.id.photo);

  imageView.setImageBitmap(BitmapFactory.decodeFile(file.getAbsolutePath()));
 
}

Using onActivityResult, tmpFile can only be declared as a class variable:

private static final int CAPTURE_IMAGE_ACTIVITY_REQUEST_CODE = 100; File tempFile;  @Override public void sendPhoto() {

Intent intentPhoto = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

ComponentName componentName = intentPhoto.resolveActivity(getPackageManager());

if (componentName != null) {

tempFile = FileUtil.getTempFile(FileUtil.FileType.IMG);

  if (tempFile != null) {

 fileUri = FileProvider.getUriForFile(this,

  BuildConfig.APPLICATION_ID + ".provider",

  tempFile);

 grantUriPermission(componentName.getPackageName(), fileUri,

  Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

  
}

  intentPhoto.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

  intentPhoto.putExtra(MediaStore.EXTRA_OUTPUT, fileUri);

  startActivityForResult(intentPhoto, CAPTURE_IMAGE_ACTIVITY_REQUEST_CODE);

}
 
}
  @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) {

if (requestCode == CAPTURE_IMAGE_ACTIVITY_REQUEST_CODE) {

  if (resultCode == RESULT_OK && fileUri != null) {

 showImagePreview(tempFile.getAbsolutePath());

  
}

}
 
}
 

@Builder

@Builder annotate Activity Class:

@Builder public class ${
ActivityName
}
 extends AppCompatActivity 

ActivityBuilder will generate two class ${ ActivityName } Builder and ${ ActivityName } Helper in same package of the activity.

using ${ ActivityName } Builder

Builder has three roles:

  • Recieve the parameters of the callee Activity
  • Setup ActivityForResult callback
  • Start Activity

First using ${ ActivityName } Builder#create to get a Activity Builder instance. In default ${ ActivityName } Builder has some callback method:

  • forCancel(Consumer<Intent>) call when RESULT_CANCEL
  • forOk(Consumer<Intent>) call when RESULT_OK
  • result(BiConsumer<Integer, Intent>) same as onActivityResult

Be carefull that the Consumer is difference from RxJava's Consumer, which Intent can be null.

And then call start to start the callee activity, start will call startActivty or startActivityForResult depend on has result callback or not. start must call from ui thread.

using ${ ActvityName } Helper

Helper has two roles:

  • Inject the parameters to the target Activity
  • Offer some convenient methods to setup result data

${ ActivityName } Helper is used by the callee activity(@Builder annotated activity)?All helper method is package, using BuildUtil.createHelper(ActivityName) to create a ${ ActivityName } Helper instance.

@Builder public class ${
ActivityName
}
 extends AppCompatActivity {

EditorActivityHelper mHelper;
@Override
protected void onCreate(Bundle savedInstanceState) {

  super.onCreate(savedInstanceState);

  mHelper = BuilderUtil.createHelper(this);

}
 
}
 

BuilderUtil.createHelper will do the injection, and must call after onCreate because the process of injection need the instacne of Intent. If you want to delay the injection, You can using new directly instance helper and call inject manually.

there are two special helper method in ${ ActivityName } Helper:

  • save(Bundle) call in Activity#onSaveInstanceState to save the parameters what need to save.
  • restore(Bundle) restore the saved parameters

more detail in keep

@BuilderParameter

@BuilderParameter work with any type of field in Activity, of cause those fields can't be private becase of these fields will inject from outside.

@Builder public class ${
ActivityName
}
 extends AppCompatActivity {

@BuilderParameter String title; // can't no be private 
}
 

each BuilderParameter ${ ActivityName } Builder will generate a corresponding setter method that support chaining:

public ${
ActivityName
}
Builder<A> title(String title) {

getIntent().putExtra("title");

return this; 
}
 

how to support any type?

We know Intent can only deliver some special type(primitive, string etc...), But the default strategy of BuilderParameter is: if the type can deliver by intent than use intent else deliver the reference.

you can change this strategy by configure transmit. for example to force the String deliver by reference:

@Builder public class ${
ActivityName
}
 extends AppCompatActivity {

@BuilderParameter(transmit = TransmitType.Ref) String title;
@BuilderParameter Object obj; 
}
 

Corresponding setter method:

public ${
ActivityName
}
Builder<A> title(String title) {

getRefMap().put("title",title);

return this; 
}

 public ${
ActivityName
}
Builder<A> obj(Object obj) {

getRefMap().put("obj", obj);

return this; 
}
 

key

By default, the key used by BuilderParameter is its variable name. key will not be exposed to the caller, but if there is a conflict, you can use key to configure the key to other name.

keep

keep, represent the parameter will be saved in Helper#save and be restored in Helper#restore. default is false, And keep only support the type that Bundle supported.

@Result

@Result annotation is used to describe the result data type of Activity(corresponding to a Result Colde).

@Result can be used for the Activity class, for methods, both ways can achieve the same purpose, the following is annotating class:

@Builder @Result(name = "content", parameters = {
 @ResultParameter(name = "content", type = String.class) 
}
) public class EditorActivity extends AppCompatActivity {
 
}
 

and @Result annotated method:

@Builder public class EditorActivity extends AppCompatActivity {
 @Result void resultContent(String content){

}
 
}
 

both two way the final code processor generated is same. the naming of @Result method need to match the regex: result(?<name>[A-Z][\w]*), as the above method content will treated as the name of the Result.

Why @Result has two kinds of usage, the main reason is that you can not use annotations to represent the parameterize type of primitive type, so only through the method statement to achieve the purpose:

@Builder public class EditorActivity extends AppCompatActivity {
 @Result void resultSelected(int index, ArrayList<User> data){

}
 
}
 

Method body can be empty or not-empty, such as wrap mHelper.resultSelected(index, data). The annotation processor dose not care about the implementation of the method, only the declaration of the method is resolved.

Helper

For each Result, the helper will generate two methods for one constant:

public class EditorActivityHelper { public static final int RESULT_CONTENT = Activity.RESULT_FIRST_USER + 1; ... void resultContent(String content) { Intent intent = new Intent(); intent.putExtra("content",content); activity.setResult(RESULT_CONTENT,intent); }

  void finishContent(String content) {

  resultContent(content);

  activity.finish();

}
 
}
 

And then in the Activity can be used:

@Override public boolean onOptionsItemSelected(MenuItem item) {

switch (item.getItemId()) {

  case R.id.action_ok:

 // set the content to result and finish activity

 mHelper.finishContent(mBinding.editText.getText().toString());

 return true;

}

... 
}
 

Builder

For Builder, each Result also generates two methods:

public class EditorActivityBuilder<A extends Activity> extends BaseActivityBuilder<EditorActivityBuilder<A>, A>{

...
public EditorActivityBuilder<A> forContent(Consumer<String> contentConsumer) {

  getConsumer().contentConsumer = (activity, content) -> contentConsumer.accept(content);

  return this;

}

public EditorActivityBuilder<A> forContent(BiConsumer<A, String> contentConsumer) {

  getConsumer().contentConsumer = contentConsumer;
  return this;

}

... 
}
 

Then you can use EditorActivityBuilder.create(this).forContent(System.out::println).start(), one line of code to start the Actiivty and handle the callback.

Type parameter A is the instance of the caller Activity reference, why should there be tow callbacks see Lambda Reference Problem

Result Parameter

Each Result can have one or more parameters or no parameter, for example:

@Builder @Result(name = "delete") public class UserDetailActivity extends AppCompatActivity {
 
}
 

Corresponding Builder method?

public UserDetailBuilder<A> forDelete(Runnable deleteConsumer) 

Result support multi parameters, but comes with only 3 callback, are Consumer, BiConsumer, TriConsumer. If the number of parameters exceeds the built-in Consumer, the annotation processor will automatically careate a new Consumer.

@Result public void resultAbcd(String a, String b, String c, String d)  

The new Consumer ActivityBuilder created:

package info.dourok.esactivity.function;  public interface Consumer4<T0, T1, T2, T3> {

void accept(T0 t0, T1 t1, T2 t2, T3 t3);
 
}
 

and

package info.dourok.esactivity.function;  public interface Consumer5<T0, T1, T2, T3, T4> {

void accept(T0 t0, T1 t1, T2 t2, T3 t3, T4 t4);
 
}
 

Corresponding Builder method?

public ${
ActivityName
}
Builder<A> forAbcd(Consumer4<String, String, String, String> abcdConsumer) {
...
}
 public ${
ActivityName
}

}
Builder<A> forAbcd(Consumer5<A, String, String, String, String> abcdConsumer) {
...
}
 

TransmitType

As with @ BuilderParameter, by default, the type is supported by Intent will passed through Intent, other objects pass the reference directly. But you can also configure a different TransmitType, for the Result method, the need to introduce a new annotation @ Transmit to configure the method parameters

@Result public void resultText(@Transmit(TransmitType.REF) String name){

}
 

Multiple Result

An Activity can have multiple Result, with the method statement with @Result annotation can easily achieve multiple Result, as long as the method name is not the same. But @Result annotation class, lower than java 8, is not the same target using multiple of the same annotation, then you can use @ResultSet to achieve:

@Builder @ResultSet(results = {

 @Result(name = "date",parameters = {
@ResultParameter(name = "date", type = Long.class)
}
),

@Result(name = "text",parameters = {

  @ResultParameter(name = "ids", type = ArrayList.class),

  @ResultParameter(name = "name", type = Character.class)
}
)
}
) public class SomeActivity extends AppCompatActivity {
  
}
 

Of course, the method way is more concise:

@Result void resultDate(Long date){

}
 @Result void resultText(ArrayList ids, Character name){

}
 

Lambda Reference Problem

ActivityBuilder is not recommended for direct use in the caller Activity, more recommended for MVP Presenter, or MVVM's ViewModel. The best practice is to combine with Android Architecture Components and Databinding.

ActivityBuilder using lambda expression to save the callback. In the internal implementation, these lambda expression is stored in a retain instance framgent. If the lambda expression is declared in the caller activity, be careful because the lambda is likely to capture a reference of the caller activity, which means that when activity is rebuilt because of some configuration changed, there is a retain instance fragment that indirectly holds the reference to the Activity to be destroyed.

But this will not lead to serious memory leak problems, because our retain instance fragment will always release the reference to the lambda expression, more serious in this case lambda expression will be executed in the wrong state, because it captures the variables It is probably a variable that has been delcared in deprecated Activity.

To avoid this, the ideal is to use a stateless lambda expression. However, our function interface is Consumer, in general Consumer always have some side effects, because it receives the parameters and then no return. Or avoid captures this.

For lambda expressions the following cases will capture this:

  • direct reference to the instance field of Activity
  • calls the instance method
  • used the this reference
  • used the super reference

For method references:

  • this keyword method reference
  • super keyword method reference
  • Constructor references for non-static internal classes
  • Activity or its instance field variable arguments method reference

the situation is still a lot, so this is why each Result has to generate two callback method, if the lambda expression need to capture of the Activity reference, please use another callback replaced:

EditorActivityBuilder.create(this)

  .forContent(text -> showToast(this,text));
 

Replace with:

EditorActivityBuilder.create(this)

  .forContent((activity,text) -> showToast(activity,text));
 

This way is an improvement not a solution, especially related to the situation about update of the View. It is recommended to use the ViewModel and Databinding to define lambda and implement updates to View.

Resources

The Firebase Android JobDispatcher is a library that provides a high-level wrapper around job scheduling engines on Android, starting with the GCM Network Manager.

Sticky observable nested ScrollView for Android.

Lib that imports all the vector drawables from materialdesignicons.com into your R.drawable/.

A simple plain pie chart widget fully customizable.

Library for toolbar animation.

A simple ViewPager indicator implementation compatible with the Android Support Library. It can use arrows on the left and on right and it can display a pageIndicator.

Topics


2D Engines   3D Engines   9-Patch   Action Bars   Activities   ADB   Advertisements   Analytics   Animations   ANR   AOP   API   APK   APT   Architecture   Audio   Autocomplete   Background Processing   Backward Compatibility   Badges   Bar Codes   Benchmarking   Bitmaps   Bluetooth   Blur Effects   Bread Crumbs   BRMS   Browser Extensions   Build Systems   Bundles   Buttons   Caching   Camera   Canvas   Cards   Carousels   Changelog   Checkboxes   Cloud Storages   Color Analysis   Color Pickers   Colors   Comet/Push   Compass Sensors   Conferences   Content Providers   Continuous Integration   Crash Reports   Credit Cards   Credits   CSV   Curl/Flip   Data Binding   Data Generators   Data Structures   Database   Database Browsers   Date &   Debugging   Decompilers   Deep Links   Dependency Injections   Design   Design Patterns   Dex   Dialogs   Distributed Computing   Distribution Platforms   Download Managers   Drawables   Emoji   Emulators   EPUB   Equalizers &   Event Buses   Exception Handling   Face Recognition   Feedback &   File System   File/Directory   Fingerprint   Floating Action   Fonts   Forms   Fragments   FRP   FSM   Functional Programming   Gamepads   Games   Geocaching   Gestures   GIF   Glow Pad   Gradle Plugins   Graphics   Grid Views   Highlighting   HTML   HTTP Mocking   Icons   IDE   IDE Plugins   Image Croppers   Image Loaders   Image Pickers   Image Processing   Image Views   Instrumentation   Intents   Job Schedulers   JSON   Keyboard   Kotlin   Layouts   Library Demos   List View   List Views   Localization   Location   Lock Patterns   Logcat   Logging   Mails   Maps   Markdown   Mathematics   Maven Plugins   MBaaS   Media   Menus   Messaging   MIME   Mobile Web   Native Image   Navigation   NDK   Networking   NFC   NoSQL   Number Pickers   OAuth   Object Mocking   OCR Engines   OpenGL   ORM   Other Pickers   Parallax List   Parcelables   Particle Systems   Password Inputs   PDF   Permissions   Physics Engines   Platforms   Plugin Frameworks   Preferences   Progress Indicators   ProGuard   Properties   Protocol Buffer   Pull To   Purchases   Push/Pull   QR Codes   Quick Return   Radio Buttons   Range Bars   Ratings   Recycler Views   Resources   REST   Ripple Effects   RSS   Screenshots   Scripting   Scroll Views   SDK   Search Inputs   Security   Sensors   Services   Showcase Views   Signatures   Sliding Panels   Snackbars   SOAP   Social Networks   Spannable   Spinners   Splash Screens   SSH   Static Analysis   Status Bars   Styling   SVG   System   Tags   Task Managers   TDD &   Template Engines   Testing   Testing Tools   Text Formatting   Text Views   Text Watchers   Text-to   Toasts   Toolkits For   Tools   Tooltips   Trainings   TV   Twitter   Updaters   USB   User Stories   Utils   Validation   Video   View Adapters   View Pagers   Views   Watch Face   Wearable Data   Wearables   Weather   Web Tools   Web Views   WebRTC   WebSockets   Wheel Widgets   Wi-Fi   Widgets   Windows   Wizards   XML   XMPP   YAML   ZIP Codes