EasyActivityResult:startActivityForResult的优雅实现

在平时的android开发中,难免少不了使用startActivityForResult去跳转到另一个Activity获取返回值,但是这种方式存在一定的不足,不够好用,为什么不好呢?让我们先来举个例子:

  • 使用系统相机拍照:
public class MainActivity extends AppCompatActivity {

    ImageView image;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        image = findViewById(R.id.image);
        findViewById(R.id.takePictures).setOnClickListener(v -> takePictures());
    }

    /**
     * 使用系统相机拍照
     */
    private void takePictures() {
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        startActivityForResult(intent, 1);
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if(requestCode == 1){ // 获取拍照结果
            Bundle bundle = data.getExtras(); // 从data中取出传递回来缩略图的信息,图片质量差,适合传递小图片
            Bitmap bitmap = (Bitmap) bundle.get("data"); // 将data中的信息流解析为Bitmap类型
            image.setImageBitmap(bitmap);
        }
    }
}
  • 布局
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <Button
        android:id="@+id/takePictures"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="拍照"
        app:layout_constraintBottom_toTopOf="@id/image"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <ImageView
        android:id="@+id/image"
        app:layout_constraintTop_toBottomOf="@id/takePictures"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintBottom_toBottomOf="parent"
        android:layout_width="200dp"
        android:layout_height="300dp"/>
</androidx.constraintlayout.widget.ConstraintLayout>

在上面这个例子中,我们设置了一个按钮,点击后使用startActivityForResult跳转到系统相机进行拍照,然后在onActivityResult中获取拍照后的图片进行显示。咋一看好像没什么问题,但实际上我们的逻辑往往没有这么简单,这时候产品经理给我们加了一个需求,要求可以拍摄视频,那我们再改一下,加一个录像按钮:

public class MainActivity extends AppCompatActivity {

    ImageView image;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        image = findViewById(R.id.image);
        findViewById(R.id.takePictures).setOnClickListener(v -> takePictures());
        findViewById(R.id.takeVideos).setOnClickListener(v -> takeVideos());
    }

    /**
     * 使用系统相机拍照
     */
    private void takePictures() {
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        startActivityForResult(intent, 1);
    }

	/**
	* 使用系统相机录像
	*/
    private void takeVideos(){
        Intent intent=new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
        startActivityForResult(intent, 2);
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if(requestCode == 1){ // 获取拍照结果
            Bundle bundle = data.getExtras(); // 从data中取出传递回来缩略图的信息,图片质量差,适合传递小图片
            Bitmap bitmap = (Bitmap) bundle.get("data"); // 将data中的信息流解析为Bitmap类型
            image.setImageBitmap(bitmap);
        }else if(requestCode == 2){ // 获取录像结果
            Uri uri = data.getData(); // 视频的uri
            // 处理逻辑
        }
    }
}
  • 布局
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <Button
        android:id="@+id/takePictures"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="拍照"
        app:layout_constraintBottom_toTopOf="@id/takeVideos"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <Button
        android:id="@+id/takeVideos"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="录像"
        app:layout_constraintTop_toBottomOf="@id/takePictures"
        app:layout_constraintBottom_toTopOf="@id/image"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"/>

    <ImageView
        android:id="@+id/image"
        app:layout_constraintTop_toBottomOf="@id/takeVideos"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintBottom_toBottomOf="parent"
        android:layout_width="200dp"
        android:layout_height="300dp"/>
</androidx.constraintlayout.widget.ConstraintLayout>

现在我们可以看到,我们每一个请求都要定义一个唯一的请求码,而且处理返回值都是在onActivityResult中进行的,一旦项目越来越复杂,需要定义更多的请求码,并且在onActivityResult中的逻辑也会越来越复杂,可读性会大大降低,项目会更加难以维护,那么有没有更优雅的解决办法呢,有,就是使用Fragment进行代理,Activitiy中可以做到的在Fragment中基本都可以实现,通过向FragmentManager中add一个没有视图的Fragment,那我们就只需要操控这个Fragment就行了。为此我封装成了一个库,方便大家使用,地址在这https://github.com/offile/EasyActivityResult

使用起来也是非常简单:

  1. 首先在顶部的build.gradle文件中添加jitpack仓库

    allprojects {
      repositories {
        ...
        // 添加下面这一行
        maven { url 'https://jitpack.io' }
      }
    }
    
  2. 接着在项目的build.gradle中添加依赖

    dependencies {
        implementation "com.github.offile:EasyActivityResult:0.9.9-beta"
    }
    
  3. 在Activity或者Fragment中使用

    // 首先创建EasyActivityResult实例(this可以是FragmentActivity或者Framgnet)
    EasyActivityResult easyActivityResult = EasyActivityResult.with(this);
    // 创建intent
    Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    // 像调用Activity的startActivityForResult一样,不用请求码,只要一个回调用来接收结果
    easyActivityResult.startActivityForResult(intent, (resultCode, intent) -> {
    	// 在这里处理结果
    });
    

看起来是不是很简单,现在让我们用这个库改写一下上面的例子:

public class MainActivity extends AppCompatActivity {

    ImageView image;
    
    // EasyActivityResult实例
    EasyActivityResult easyActivityResult = EasyActivityResult.with(this);

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        image = findViewById(R.id.image);
        findViewById(R.id.takePictures).setOnClickListener(v -> takePictures());
        findViewById(R.id.takeVideos).setOnClickListener(v -> takeVideos());
    }

    /**
     * 使用系统相机拍照
     */
    private void takePictures() {
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        easyActivityResult.startActivityForResult(intent, (i, data) -> {
            Bundle bundle = data.getExtras(); // 从data中取出传递回来缩略图的信息,图片质量差,适合传递小图片
            Bitmap bitmap = (Bitmap) bundle.get("data"); // 将data中的信息流解析为Bitmap类型
            image.setImageBitmap(bitmap);
        });
    }

    private void takeVideos(){
        Intent intent=new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
        easyActivityResult.startActivityForResult(intent, (i, data) -> {
            Uri uri = data.getData(); // 视频的uri
        });
    }
}

现在我们可以看到,不用再重写onActivityResult方法了,烦人的请求码也没有了,可读性也提高了很多,也不用担心代码越来越多的问题了。

需要查看详细源码和demo,请移步我的github,欢迎star和fork

点个赞吧,你的鼓励就是我前进的动力

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值