需求:通过点击拍照按钮来调用手机相机,拍照后显示在屏幕中
1.布局文件设计
xml文件中放置一个按钮以及一个imageView控件(图片放置于此)
PS:个人推荐直接用约束布局拖组件
代码如下:
<?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">
<Button
android:id="@+id/take_photo"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="拍照"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintHorizontal_bias="0.498"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintVertical_bias="0.109" />
<ImageView
android:id="@+id/picture"
android:layout_width="0dp"
android:layout_height="0dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintHorizontal_bias="0.461"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="@+id/guideline2"
tools:srcCompat="@drawable/baseline_child_care_24" />
<androidx.constraintlayout.widget.Guideline
android:id="@+id/guideline2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="horizontal"
app:layout_constraintGuide_percent="0.5" />
</androidx.constraintlayout.widget.ConstraintLayout>
2.授予权限
由于我们实现的功能是在公共存储读写数据,就必须使用“WRITE_EXTERNAL_STORAGE”权限,也就要去掉“ android:maxSdkVersion="18"”。
在manifest文件中授权
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
android:maxSdkVersion="18"/>
3.使用FIleProvider类
由于我们的需求需要在我们当前应用程序和相机之间共享数据(即所拍照片),我们需要用到FileProvider类
配置过程:
首先需要在
AndroidManifest.xml
文件中通过<provider>
标签进行声明,并配置自定义的android:authorities
属性,代码如下:
<manifest>
<application>
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="my.itgungnir.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
</provider>
</application>
</manifest>
如果你的模拟器版本在Android10之下,就用android.support.v4.content.FileProvider
为了让其他应用可以访问当前应用下的文件,我们还需要配置哪些文件夹可以被访问,这个步骤也是在XML
文件中配置的。我们需要在项目的/res/xml
文件夹下创建一个路径配置文件,命名为file_paths.xml
(文件名可以自定义),这个文件中的根节点是<paths>
,在这个节点下配置文件夹。一个示例配置如下:
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path name="my_images" path="/"/>
</paths>
path路径设为“/”,表示能够访问所有文件夹
配置完各个文件夹之后,我们还需要在AndroidManifest.xml
文件中的<provider>
标签中引用这个配置,这样才能使配置生效。使用<meta-data>
标签来引用配置,代码如下:
<provider
android:authorities="Tom_LZK"
android:name="androidx.core.content.FileProvider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths"/>
</provider>
<meta-data>
标签中的android:name
属性是固定值android.support.FILE_PROVIDER_PATHS
,android:resource
属性是对上面的配置文件的引用。
附上MainActivity的代码:
package com.example.arbitrary;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Bundle;
import android.provider.MediaStore;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.content.FileProvider;
import java.io.File;
import java.io.IOException;
public class MainActivity extends AppCompatActivity {
public static final int TAKE_PHOTO = 1;
private ImageView picture;
private Uri imageUri;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button takePhoto = findViewById(R.id.take_photo);
picture = findViewById(R.id.picture);
takePhoto.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
//创建File对象,用来存储拍照后的照片
//getExternalCacheDir()获取此应用缓存数据的位置,在这个位置保存图片
File outputImage = new File(getExternalCacheDir(), "output_image.jpg");
try {
if (outputImage.exists()) {//如果图片已经存在就删除再重新创建
outputImage.delete();
}
outputImage.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
imageUri = FileProvider.getUriForFile(MainActivity.this,
"Tom_LZK", outputImage);
//启动相机
Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent, TAKE_PHOTO);
}
});
}
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == TAKE_PHOTO && resultCode == RESULT_OK) {
//显示拍出来的照片
Bitmap bitmap = BitmapFactory.decodeFile(getExternalCacheDir()+"//output_image.jpg");
picture.setImageBitmap(bitmap);
}
}
}
完成。
注意:getUriForFile方法中authorities变量要与provider中的authorities一致