在平时的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
使用起来也是非常简单:
-
首先在顶部的build.gradle文件中添加jitpack仓库
allprojects { repositories { ... // 添加下面这一行 maven { url 'https://jitpack.io' } } }
-
接着在项目的build.gradle中添加依赖
dependencies { implementation "com.github.offile:EasyActivityResult:0.9.9-beta" }
-
在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
点个赞吧,你的鼓励就是我前进的动力